index.vue 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. <template>
  2. <view class="van-stepper custom-class">
  3. <view
  4. v-if="showMinus"
  5. data-type="minus"
  6. :style="'width: ' + utils.addUnit(buttonSize) + '; height: ' + utils.addUnit(buttonSize)"
  7. :class="'minus-class ' + utils.bem('stepper__minus', { disabled: disabled || disableMinus || currentValue <= min })"
  8. hover-class="van-stepper__minus--hover"
  9. hover-stay-time="70"
  10. @tap="onTap"
  11. @touchstart="onTouchStart"
  12. @touchend="onTouchEnd"
  13. />
  14. <input
  15. :type="integer ? 'number' : 'digit'"
  16. :class="'input-class ' + utils.bem('stepper__input', { disabled: disabled || disableInput })"
  17. :style="'width: ' + utils.addUnit(inputWidth) + '; height: ' + utils.addUnit(buttonSize)"
  18. :value="currentValue"
  19. :focus="focus"
  20. :disabled="disabled || disableInput"
  21. @input="onInput"
  22. @focus="onFocus"
  23. @blur="onBlur"
  24. />
  25. <view
  26. v-if="showPlus"
  27. data-type="plus"
  28. :style="'width: ' + utils.addUnit(buttonSize) + '; height: ' + utils.addUnit(buttonSize)"
  29. :class="'plus-class ' + utils.bem('stepper__plus', { disabled: disabled || disablePlus || currentValue >= max })"
  30. hover-class="van-stepper__plus--hover"
  31. hover-stay-time="70"
  32. @tap="onTap"
  33. @touchstart="onTouchStart"
  34. @touchend="onTouchEnd"
  35. />
  36. </view>
  37. </template>
  38. <script module="utils" lang="wxs" src="@/miniprogram_npm/@vant/weapp/wxs/utils.wxs"></script>
  39. <script>
  40. 'use strict';
  41. var __assign =
  42. (this && this.__assign) ||
  43. function () {
  44. __assign =
  45. Object.assign ||
  46. function (t) {
  47. for (var s, i = 1, n = arguments.length; i < n; i++) {
  48. s = arguments[i];
  49. for (var p in s) {
  50. if (Object.prototype.hasOwnProperty.call(s, p)) {
  51. t[p] = s[p];
  52. }
  53. }
  54. }
  55. return t;
  56. };
  57. return __assign.apply(this, arguments);
  58. };
  59. Object.defineProperty(exports, '__esModule', {
  60. value: true
  61. });
  62. var component_1 = require('../common/component');
  63. var utils_1 = require('../common/utils');
  64. var LONG_PRESS_START_TIME = 600;
  65. var LONG_PRESS_INTERVAL = 200;
  66. // add num and avoid float number
  67. function add(num1, num2) {
  68. var cardinal = Math.pow(10, 10);
  69. return Math.round((num1 + num2) * cardinal) / cardinal;
  70. }
  71. function equal(value1, value2) {
  72. return String(value1) === String(value2);
  73. }
  74. component_1.VantComponent({
  75. field: true,
  76. classes: ['input-class', 'plus-class', 'minus-class'],
  77. props: {
  78. value: {
  79. type: null,
  80. observer: function (value) {
  81. if (!equal(value, this.currentValue)) {
  82. this.setData({
  83. currentValue: this.format(value)
  84. });
  85. }
  86. }
  87. },
  88. integer: {
  89. type: Boolean,
  90. observer: 'check'
  91. },
  92. disabled: Boolean,
  93. inputWidth: null,
  94. buttonSize: null,
  95. asyncChange: Boolean,
  96. disableInput: Boolean,
  97. decimalLength: {
  98. type: Number,
  99. value: null,
  100. observer: 'check'
  101. },
  102. min: {
  103. type: null,
  104. value: 1,
  105. observer: 'check'
  106. },
  107. max: {
  108. type: null,
  109. value: Number.MAX_SAFE_INTEGER,
  110. observer: 'check'
  111. },
  112. step: {
  113. type: null,
  114. value: 1
  115. },
  116. showPlus: {
  117. type: Boolean,
  118. value: true
  119. },
  120. showMinus: {
  121. type: Boolean,
  122. value: true
  123. },
  124. disablePlus: Boolean,
  125. disableMinus: Boolean,
  126. longPress: {
  127. type: Boolean,
  128. value: true
  129. }
  130. },
  131. data: {
  132. currentValue: ''
  133. },
  134. created: function () {
  135. this.setData({
  136. currentValue: this.format(this.value)
  137. });
  138. },
  139. methods: {
  140. check: function () {
  141. var val = this.format(this.currentValue);
  142. if (!equal(val, this.currentValue)) {
  143. this.setData({
  144. currentValue: val
  145. });
  146. }
  147. },
  148. isDisabled: function (type) {
  149. if (type === 'plus') {
  150. return this.disabled || this.disablePlus || this.currentValue >= this.max;
  151. }
  152. return this.disabled || this.disableMinus || this.currentValue <= this.min;
  153. },
  154. onFocus: function (event) {
  155. this.$emit('focus', event.detail);
  156. },
  157. onBlur: function (event) {
  158. var value = this.format(event.detail.value);
  159. this.emitChange(value);
  160. this.$emit(
  161. 'blur',
  162. __assign(__assign({}, event.detail), {
  163. value: value
  164. })
  165. );
  166. },
  167. // filter illegal characters
  168. filter: function (value) {
  169. value = String(value).replace(/[^0-9.-]/g, '');
  170. if (this.integer && value.indexOf('.') !== -1) {
  171. value = value.split('.')[0];
  172. }
  173. return value;
  174. },
  175. // limit value range
  176. format: function (value) {
  177. value = this.filter(value);
  178. // format range
  179. value = value === '' ? 0 : +value;
  180. value = Math.max(Math.min(this.max, value), this.min);
  181. // format decimal
  182. if (utils_1.isDef(this.decimalLength)) {
  183. value = value.toFixed(this.decimalLength);
  184. }
  185. return value;
  186. },
  187. onInput: function (event) {
  188. var _a = (event.detail || {}).value;
  189. var value = _a === void 0 ? '' : _a; // allow input to be empty
  190. if (value === '') {
  191. return;
  192. }
  193. var formatted = this.filter(value);
  194. // limit max decimal length
  195. if (utils_1.isDef(this.decimalLength) && formatted.indexOf('.') !== -1) {
  196. var pair = formatted.split('.');
  197. formatted = pair[0] + '.' + pair[1].slice(0, this.decimalLength);
  198. }
  199. this.emitChange(formatted);
  200. },
  201. emitChange: function (value) {
  202. if (!this.asyncChange) {
  203. this.setData({
  204. currentValue: value
  205. });
  206. }
  207. this.$emit('change', value);
  208. },
  209. onChange: function () {
  210. var type = this.type;
  211. if (this.isDisabled(type)) {
  212. this.$emit('overlimit', type);
  213. return;
  214. }
  215. var diff = type === 'minus' ? -this.step : +this.step;
  216. var value = this.format(add(+this.currentValue, diff));
  217. this.emitChange(value);
  218. this.$emit(type);
  219. },
  220. longPressStep: function () {
  221. var that = this;
  222. this.longPressTimer = setTimeout(function () {
  223. that.onChange();
  224. that.longPressStep();
  225. }, LONG_PRESS_INTERVAL);
  226. },
  227. onTap: function (event) {
  228. var type = event.currentTarget.dataset.type;
  229. this.type = type;
  230. this.onChange();
  231. },
  232. onTouchStart: function (event) {
  233. var that = this;
  234. if (!this.longPress) {
  235. return;
  236. }
  237. clearTimeout(this.longPressTimer);
  238. var type = event.currentTarget.dataset.type;
  239. this.type = type;
  240. this.isLongPress = false;
  241. this.longPressTimer = setTimeout(function () {
  242. that.isLongPress = true;
  243. that.onChange();
  244. that.longPressStep();
  245. }, LONG_PRESS_START_TIME);
  246. },
  247. onTouchEnd: function () {
  248. if (!this.longPress) {
  249. return;
  250. }
  251. clearTimeout(this.longPressTimer);
  252. }
  253. }
  254. });
  255. </script>
  256. <style>
  257. @import './index.css';
  258. </style>