tui-textarea.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568
  1. <template>
  2. <view :class="{'tui-textarea__border':textareaBorder}" :style="{marginTop:marginTop+'rpx'}" @tap="fieldClick">
  3. <view class="tui-textarea__wrap"
  4. :class="{'tui-line__left':lineLeft,'tui-border__top':!borderTop || textareaBorder,'tui-border__bottom':!borderBottom || textareaBorder,'tui-textarea__flex-start':flexStart}"
  5. :style="{padding:padding,backgroundColor:backgroundColor}">
  6. <!-- #ifdef APP-NVUE -->
  7. <view class="tui-textarea__required" v-if="required && !flexStart">
  8. <text :style="{color:requiredColor,paddingTop:'2rpx'}">*</text>
  9. </view>
  10. <text class="tui-textarea__required" :style="{color:requiredColor,top:requiredTop}"
  11. v-if="required && flexStart">*</text>
  12. <!-- #endif -->
  13. <!-- #ifndef APP-NVUE -->
  14. <view class="tui-textarea__required" :class="{'tui-required__flex-start':flexStart}"
  15. :style="{color:requiredColor,top:flexStart?requiredTop:'50%'}" v-if="required">*</view>
  16. <!-- #endif -->
  17. <view class="tui-textarea__label"
  18. :style="{fontSize:labelSize+'rpx',color:labelColor,minWidth:labelWidth+'rpx'}" v-if="label">
  19. <text :style="{fontSize:labelSize+'rpx',color:labelColor}">{{label}}</text>
  20. </view>
  21. <slot name="left"></slot>
  22. <view class="tui-textarea__flex-1">
  23. <textarea class="tui-textarea__self" :class="{'tui-text__right':textRight}"
  24. :style="{height:height,minHeight:minHeight,fontSize:size+'rpx',color:color}"
  25. placeholder-class="tui-placeholder" :name="name" :value="inputVal" :placeholder="placeholder"
  26. :placeholderStyle="placeholderStyl" :disabled="disabled" :cursor-spacing="cursorSpacing"
  27. :maxlength="maxlength" :focus="focused" :auto-height="autoHeight" :fixed="fixed"
  28. :show-confirm-bar="showConfirmBar" :cursor="cursor" :selection-start="selectionStart"
  29. :selection-end="selectionEnd" :adjust-position="adjustPosition" :hold-keyboard="holdKeyboard"
  30. :show-count="false" :disable-default-padding="disableDefaultPadding" @focus="onFocus" @blur="onBlur"
  31. @input="onInput" @confirm="onConfirm" @linechange="onLinechange"
  32. @keyboardheightchange="onKeyboardheightchange"></textarea>
  33. <view class="tui-textarea__counter" :style="{fontSize:counterSize+'rpx',color:counterColor}"
  34. v-if="isCounter">
  35. <text
  36. :style="{fontSize:counterSize+'rpx',color:counterColor}">{{maxlength!=-1?`${count}/${maxlength}`:count}}</text>
  37. </view>
  38. </view>
  39. <slot name="right"></slot>
  40. </view>
  41. </view>
  42. </template>
  43. <script>
  44. export default {
  45. name: "tui-textarea",
  46. emits: ['input', 'update:modelValue', 'focus', 'blur', 'confirm', 'click', 'linechange', 'keyboardheightchange'],
  47. //这里加group是为了避免在表单中使用时给组件加value属性
  48. // #ifndef VUE3
  49. // #ifdef MP-WEIXIN
  50. //加group是为了避免在表单中使用时给组件加value属性
  51. behaviors: ['wx://form-field-group'],
  52. // #endif
  53. // #ifdef MP-BAIDU || MP-QQ
  54. //如果在这些平台不需要也能识别,则删除
  55. behaviors: ['uni://form-field'],
  56. // #endif
  57. // #endif
  58. // #ifdef MP-WEIXIN
  59. options: {
  60. addGlobalClass: true,
  61. virtualHost: true
  62. },
  63. // #endif
  64. props: {
  65. //是否为必填项
  66. required: {
  67. type: Boolean,
  68. default: false
  69. },
  70. requiredColor: {
  71. type: String,
  72. default: '#EB0909'
  73. },
  74. requiredTop: {
  75. type: String,
  76. default: '32rpx'
  77. },
  78. //左侧标题
  79. label: {
  80. type: String,
  81. default: ''
  82. },
  83. //标题字体大小
  84. labelSize: {
  85. type: Number,
  86. default: 32
  87. },
  88. labelColor: {
  89. type: String,
  90. default: '#333'
  91. },
  92. //label 最小宽度 rpx
  93. labelWidth: {
  94. type: Number,
  95. default: 140
  96. },
  97. //获取焦点
  98. focus: {
  99. type: Boolean,
  100. default: false
  101. },
  102. autoHeight: {
  103. type: Boolean,
  104. default: false
  105. },
  106. fixed: {
  107. type: Boolean,
  108. default: false
  109. },
  110. placeholder: {
  111. type: String,
  112. default: ''
  113. },
  114. placeholderStyle: {
  115. type: String,
  116. default: ''
  117. },
  118. //输入框名称
  119. name: {
  120. type: String,
  121. default: ''
  122. },
  123. //输入框值
  124. value: {
  125. type: [Number, String],
  126. default: ''
  127. },
  128. // #ifdef VUE3
  129. //输入框值
  130. modelValue: {
  131. type: [Number, String],
  132. default: ''
  133. },
  134. // #endif
  135. disabled: {
  136. type: Boolean,
  137. default: false
  138. },
  139. maxlength: {
  140. type: [Number, String],
  141. default: 140
  142. },
  143. cursorSpacing: {
  144. type: Number,
  145. default: 0,
  146. },
  147. showConfirmBar: {
  148. type: Boolean,
  149. default: true
  150. },
  151. cursor: {
  152. type: Number,
  153. default: -1
  154. },
  155. selectionStart: {
  156. type: Number,
  157. default: -1
  158. },
  159. selectionEnd: {
  160. type: Number,
  161. default: -1
  162. },
  163. adjustPosition: {
  164. type: Boolean,
  165. default: true
  166. },
  167. disableDefaultPadding: {
  168. type: Boolean,
  169. default: true
  170. },
  171. holdKeyboard: {
  172. type: Boolean,
  173. default: false
  174. },
  175. height: {
  176. type: String,
  177. default: '200rpx'
  178. },
  179. minHeight: {
  180. type: String,
  181. default: '200rpx'
  182. },
  183. //标题与输入框是否顶端对齐
  184. flexStart: {
  185. type: Boolean,
  186. default: false
  187. },
  188. //输入框字体大小 rpx
  189. size: {
  190. type: Number,
  191. default: 32
  192. },
  193. //输入框字体颜色
  194. color: {
  195. type: String,
  196. default: '#333'
  197. },
  198. // 是否显示 textarea 边框
  199. textareaBorder: {
  200. type: Boolean,
  201. default: false
  202. },
  203. // 是否显示上边框
  204. borderTop: {
  205. type: Boolean,
  206. default: true
  207. },
  208. // 是否显示下边框
  209. borderBottom: {
  210. type: Boolean,
  211. default: true
  212. },
  213. //下边框线条是否有左偏移距离
  214. lineLeft: {
  215. type: Boolean,
  216. default: false
  217. },
  218. // 是否自动去除两端的空格
  219. trim: {
  220. type: Boolean,
  221. default: true
  222. },
  223. textRight: {
  224. type: Boolean,
  225. default: false
  226. },
  227. //输入框padding值
  228. padding: {
  229. type: String,
  230. default: '26rpx 30rpx'
  231. },
  232. //输入框背景颜色
  233. backgroundColor: {
  234. type: String,
  235. default: '#FFFFFF'
  236. },
  237. //输入框margin-top值 rpx
  238. marginTop: {
  239. type: Number,
  240. default: 0
  241. },
  242. //是否显示底部输入长度计数
  243. isCounter: {
  244. type: Boolean,
  245. default: false
  246. },
  247. //计数文本颜色
  248. counterColor: {
  249. type: String,
  250. default: '#999'
  251. },
  252. //计数文本大小 rpx
  253. counterSize: {
  254. type: Number,
  255. default: 28
  256. }
  257. },
  258. data() {
  259. return {
  260. placeholderStyl: '',
  261. count: 0,
  262. focused: false,
  263. inputVal: ''
  264. };
  265. },
  266. watch: {
  267. focus(val) {
  268. this.$nextTick(() => {
  269. this.focused = val
  270. })
  271. },
  272. placeholderStyle() {
  273. this.fieldPlaceholderStyle()
  274. },
  275. // #ifdef VUE3
  276. modelValue(newVal) {
  277. this.inputVal = newVal || ''
  278. this.count = this.getCount(this.inputVal.length)
  279. },
  280. // #endif
  281. value(newVal) {
  282. this.inputVal = newVal || ''
  283. this.count = this.getCount(this.inputVal.length)
  284. }
  285. },
  286. created() {
  287. // #ifndef VUE3
  288. this.inputVal = this.value || ''
  289. // #endif
  290. // #ifdef VUE3
  291. if (this.value && !this.modelValue) {
  292. this.inputVal = this.value || ''
  293. } else {
  294. this.inputVal = this.modelValue || ''
  295. }
  296. // #endif
  297. this.count = this.getCount(this.inputVal.length)
  298. this.fieldPlaceholderStyle()
  299. },
  300. mounted() {
  301. this.$nextTick(() => {
  302. this.focused = this.focus
  303. })
  304. },
  305. methods: {
  306. getCount(count) {
  307. const max = Number(this.maxlength)
  308. if (count > max && max !== -1) {
  309. return max
  310. }
  311. return count;
  312. },
  313. fieldPlaceholderStyle() {
  314. if (this.placeholderStyle) {
  315. this.placeholderStyl = this.placeholderStyle
  316. } else {
  317. const size = uni.upx2px(this.size)
  318. this.placeholderStyl = `fontSize:${size}px`
  319. }
  320. },
  321. onInput(event) {
  322. let value = event.detail.value;
  323. if (this.trim) value = this.trimStr(value);
  324. const lenth = value.length;
  325. const max = Number(this.maxlength)
  326. if (lenth > max && max !== -1) {
  327. lenth = max;
  328. value = value.substring(0, max - 1)
  329. }
  330. this.count = lenth;
  331. this.$emit('input', value);
  332. // #ifdef VUE3
  333. this.$emit('update:modelValue', value)
  334. // #endif
  335. },
  336. onFocus(event) {
  337. this.$emit('focus', event);
  338. },
  339. onBlur(event) {
  340. this.$emit('blur', event);
  341. },
  342. onConfirm(e) {
  343. this.$emit('confirm', e.detail.value);
  344. },
  345. fieldClick() {
  346. this.$emit('click', {
  347. name: this.name
  348. });
  349. },
  350. onLinechange(e) {
  351. this.$emit('linechange', e.detail)
  352. },
  353. onKeyboardheightchange(e) {
  354. this.$emit('keyboardheightchange', e.detail)
  355. },
  356. trimStr(str) {
  357. return str.replace(/^\s+|\s+$/g, '');
  358. }
  359. }
  360. }
  361. </script>
  362. <style>
  363. .tui-textarea__wrap {
  364. /* #ifndef APP-NVUE */
  365. width: 100%;
  366. box-sizing: border-box;
  367. display: flex;
  368. /* #endif */
  369. flex-direction: row;
  370. flex: 1;
  371. align-items: center;
  372. position: relative;
  373. /* #ifdef APP-NVUE */
  374. border-top-width: 0.5px;
  375. border-top-style: solid;
  376. border-top-color: rgba(0, 0, 0, 0.1);
  377. border-bottom-width: 0.5px;
  378. border-bottom-style: solid;
  379. border-bottom-color: rgba(0, 0, 0, 0.1);
  380. padding: 26rpx 30rpx;
  381. /* #endif */
  382. }
  383. .tui-textarea__flex-start {
  384. align-items: flex-start !important;
  385. }
  386. /* #ifndef APP-NVUE */
  387. .tui-textarea__wrap::before {
  388. content: ' ';
  389. position: absolute;
  390. top: 0;
  391. right: 0;
  392. left: 0;
  393. border-top: 1px solid var(--thorui-line-color, rgba(0, 0, 0, 0.1));
  394. -webkit-transform: scaleY(0.5);
  395. transform: scaleY(0.5);
  396. transform-origin: 0 0;
  397. z-index: 2;
  398. pointer-events: none;
  399. }
  400. .tui-textarea__wrap::after {
  401. content: ' ';
  402. position: absolute;
  403. border-bottom: 1px solid var(--thorui-line-color, rgba(0, 0, 0, 0.1));
  404. -webkit-transform: scaleY(0.5);
  405. transform: scaleY(0.5);
  406. transform-origin: 0 100%;
  407. bottom: 0;
  408. right: 0;
  409. left: 0;
  410. z-index: 2;
  411. pointer-events: none;
  412. }
  413. .tui-line__left::after {
  414. left: 30rpx !important;
  415. }
  416. .tui-border__top::before {
  417. border-top: 0;
  418. }
  419. .tui-border__bottom::after {
  420. border-bottom: 0;
  421. }
  422. /* #endif */
  423. /* #ifdef APP-NVUE */
  424. .tui-border__top {
  425. border-top-width: 0;
  426. }
  427. .tui-border__bottom {
  428. border-bottom-width: 0;
  429. }
  430. /* #endif */
  431. .tui-textarea__required {
  432. position: absolute;
  433. left: 12rpx;
  434. /* #ifndef APP-NVUE */
  435. height: 30rpx;
  436. top: 50%;
  437. transform: translateY(-50%);
  438. line-height: 1.15;
  439. /* #endif */
  440. /* #ifdef APP-NVUE */
  441. flex: 1;
  442. height: 100wx;
  443. align-items: center;
  444. justify-content: center;
  445. /* #endif */
  446. }
  447. /* #ifndef APP-NVUE */
  448. .tui-required__flex-start {
  449. transform: translateY(0);
  450. }
  451. /* #endif */
  452. .tui-textarea__label {
  453. padding-right: 12rpx;
  454. /* #ifndef APP-NVUE */
  455. flex-shrink: 0;
  456. /* #endif */
  457. }
  458. .tui-textarea__self {
  459. flex: 1;
  460. /* #ifndef APP-NVUE */
  461. width: 100%;
  462. overflow: visible;
  463. box-sizing: border-box;
  464. /* #endif */
  465. /* #ifdef APP-NVUE */
  466. padding-top: 6px;
  467. padding-bottom: 10px;
  468. /* #endif */
  469. /* #ifdef MP-ALIPAY || MP-TOUTIAO*/
  470. padding-top: 0 !important;
  471. padding-bottom: 0;
  472. /* #endif */
  473. /* #ifdef MP-TOUTIAO */
  474. background-color: rgba(255, 255, 255, 0) !important;
  475. /* #endif */
  476. }
  477. .tui-placeholder {
  478. /* #ifndef APP-NVUE */
  479. color: var(--thorui-text-color-placeholder, #ccc);
  480. overflow: visible;
  481. /* #endif */
  482. /* #ifdef APP-NVUE */
  483. color: #ccc;
  484. /* #endif */
  485. /* #ifdef MP-TOUTIAO */
  486. padding-top: 0 !important;
  487. /* #endif */
  488. }
  489. /* #ifdef MP */
  490. ::v-deep .tui-placeholder {
  491. color: var(--thorui-text-color-placeholder, #ccc);
  492. overflow: visible;
  493. }
  494. /* #endif */
  495. .tui-textarea__border {
  496. border-radius: 4rpx;
  497. position: relative;
  498. /* #ifdef APP-NVUE */
  499. border-style: solid;
  500. border-width: 0.5px;
  501. border-color: #d1d1d1;
  502. /* #endif */
  503. /* #ifndef APP-NVUE */
  504. border-width: 0;
  505. /* #endif */
  506. }
  507. /* #ifndef APP-NVUE */
  508. .tui-textarea__border::after {
  509. content: ' ';
  510. position: absolute;
  511. height: 200%;
  512. width: 200%;
  513. border: 1px solid var(--thorui-border-color, #d1d1d1);
  514. transform-origin: 0 0;
  515. transform: scale(0.5);
  516. left: 0;
  517. top: 0;
  518. border-radius: 8rpx;
  519. pointer-events: none;
  520. }
  521. /* #endif */
  522. .tui-textarea__flex-1 {
  523. flex: 1;
  524. }
  525. .tui-textarea__counter {
  526. padding-top: 8rpx;
  527. text-align: right;
  528. /* #ifdef APP-NVUE */
  529. flex-direction: row;
  530. flex: 1;
  531. justify-content: flex-end;
  532. /* #endif */
  533. }
  534. .tui-text__right {
  535. text-align: right;
  536. }
  537. </style>