| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470 | import { createTextVNode as _createTextVNode, mergeProps as _mergeProps, createVNode as _createVNode } from "vue";import { ref, watch, provide, computed, nextTick, reactive, onMounted, defineComponent } from "vue";import { isDef, extend, addUnit, toArray, FORM_KEY, numericProp, unknownProp, resetScroll, formatNumber, preventDefault, makeStringProp, makeNumericProp, createNamespace } from "../utils/index.mjs";import { cutString, runSyncRule, endComposing, mapInputType, isEmptyValue, startComposing, getRuleMessage, resizeTextarea, getStringLength, runRuleValidator } from "./utils.mjs";import { cellSharedProps } from "../cell/Cell.mjs";import { useParent, useEventListener, CUSTOM_FIELD_INJECTION_KEY } from "@vant/use";import { useId } from "../composables/use-id.mjs";import { useExpose } from "../composables/use-expose.mjs";import { Icon } from "../icon/index.mjs";import { Cell } from "../cell/index.mjs";const [name, bem] = createNamespace("field");const fieldSharedProps = {  id: String,  name: String,  leftIcon: String,  rightIcon: String,  autofocus: Boolean,  clearable: Boolean,  maxlength: numericProp,  formatter: Function,  clearIcon: makeStringProp("clear"),  modelValue: makeNumericProp(""),  inputAlign: String,  placeholder: String,  autocomplete: String,  errorMessage: String,  enterkeyhint: String,  clearTrigger: makeStringProp("focus"),  formatTrigger: makeStringProp("onChange"),  error: {    type: Boolean,    default: null  },  disabled: {    type: Boolean,    default: null  },  readonly: {    type: Boolean,    default: null  }};const fieldProps = extend({}, cellSharedProps, fieldSharedProps, {  rows: numericProp,  type: makeStringProp("text"),  rules: Array,  autosize: [Boolean, Object],  labelWidth: numericProp,  labelClass: unknownProp,  labelAlign: String,  showWordLimit: Boolean,  errorMessageAlign: String,  colon: {    type: Boolean,    default: null  }});var stdin_default = defineComponent({  name,  props: fieldProps,  emits: ["blur", "focus", "clear", "keypress", "click-input", "end-validate", "start-validate", "click-left-icon", "click-right-icon", "update:modelValue"],  setup(props, {    emit,    slots  }) {    const id = useId();    const state = reactive({      status: "unvalidated",      focused: false,      validateMessage: ""    });    const inputRef = ref();    const clearIconRef = ref();    const customValue = ref();    const {      parent: form    } = useParent(FORM_KEY);    const getModelValue = () => {      var _a;      return String((_a = props.modelValue) != null ? _a : "");    };    const getProp = (key) => {      if (isDef(props[key])) {        return props[key];      }      if (form && isDef(form.props[key])) {        return form.props[key];      }    };    const showClear = computed(() => {      const readonly = getProp("readonly");      if (props.clearable && !readonly) {        const hasValue = getModelValue() !== "";        const trigger = props.clearTrigger === "always" || props.clearTrigger === "focus" && state.focused;        return hasValue && trigger;      }      return false;    });    const formValue = computed(() => {      if (customValue.value && slots.input) {        return customValue.value();      }      return props.modelValue;    });    const runRules = (rules) => rules.reduce((promise, rule) => promise.then(() => {      if (state.status === "failed") {        return;      }      let {        value      } = formValue;      if (rule.formatter) {        value = rule.formatter(value, rule);      }      if (!runSyncRule(value, rule)) {        state.status = "failed";        state.validateMessage = getRuleMessage(value, rule);        return;      }      if (rule.validator) {        if (isEmptyValue(value) && rule.validateEmpty === false) {          return;        }        return runRuleValidator(value, rule).then((result) => {          if (result && typeof result === "string") {            state.status = "failed";            state.validateMessage = result;          } else if (result === false) {            state.status = "failed";            state.validateMessage = getRuleMessage(value, rule);          }        });      }    }), Promise.resolve());    const resetValidation = () => {      state.status = "unvalidated";      state.validateMessage = "";    };    const endValidate = () => emit("end-validate", {      status: state.status    });    const validate = (rules = props.rules) => new Promise((resolve) => {      resetValidation();      if (rules) {        emit("start-validate");        runRules(rules).then(() => {          if (state.status === "failed") {            resolve({              name: props.name,              message: state.validateMessage            });            endValidate();          } else {            state.status = "passed";            resolve();            endValidate();          }        });      } else {        resolve();      }    });    const validateWithTrigger = (trigger) => {      if (form && props.rules) {        const {          validateTrigger        } = form.props;        const defaultTrigger = toArray(validateTrigger).includes(trigger);        const rules = props.rules.filter((rule) => {          if (rule.trigger) {            return toArray(rule.trigger).includes(trigger);          }          return defaultTrigger;        });        if (rules.length) {          validate(rules);        }      }    };    const limitValueLength = (value) => {      const {        maxlength      } = props;      if (isDef(maxlength) && getStringLength(value) > maxlength) {        const modelValue = getModelValue();        if (modelValue && getStringLength(modelValue) === +maxlength) {          return modelValue;        }        return cutString(value, +maxlength);      }      return value;    };    const updateValue = (value, trigger = "onChange") => {      const originalValue = value;      value = limitValueLength(value);      const isExceedLimit = value !== originalValue;      if (props.type === "number" || props.type === "digit") {        const isNumber = props.type === "number";        value = formatNumber(value, isNumber, isNumber);      }      if (props.formatter && trigger === props.formatTrigger) {        value = props.formatter(value);      }      if (inputRef.value && inputRef.value.value !== value) {        if (state.focused && isExceedLimit) {          const {            selectionStart,            selectionEnd          } = inputRef.value;          inputRef.value.value = value;          inputRef.value.setSelectionRange(selectionStart - 1, selectionEnd - 1);        } else {          inputRef.value.value = value;        }      }      if (value !== props.modelValue) {        emit("update:modelValue", value);      }    };    const onInput = (event) => {      if (!event.target.composing) {        updateValue(event.target.value);      }    };    const blur = () => {      var _a;      return (_a = inputRef.value) == null ? void 0 : _a.blur();    };    const focus = () => {      var _a;      return (_a = inputRef.value) == null ? void 0 : _a.focus();    };    const adjustTextareaSize = () => {      const input = inputRef.value;      if (props.type === "textarea" && props.autosize && input) {        resizeTextarea(input, props.autosize);      }    };    const onFocus = (event) => {      state.focused = true;      emit("focus", event);      nextTick(adjustTextareaSize);      if (getProp("readonly")) {        blur();      }    };    const onBlur = (event) => {      if (getProp("readonly")) {        return;      }      state.focused = false;      updateValue(getModelValue(), "onBlur");      emit("blur", event);      validateWithTrigger("onBlur");      nextTick(adjustTextareaSize);      resetScroll();    };    const onClickInput = (event) => emit("click-input", event);    const onClickLeftIcon = (event) => emit("click-left-icon", event);    const onClickRightIcon = (event) => emit("click-right-icon", event);    const onClear = (event) => {      preventDefault(event);      emit("update:modelValue", "");      emit("clear", event);    };    const showError = computed(() => {      if (typeof props.error === "boolean") {        return props.error;      }      if (form && form.props.showError && state.status === "failed") {        return true;      }    });    const labelStyle = computed(() => {      const labelWidth = getProp("labelWidth");      if (labelWidth) {        return {          width: addUnit(labelWidth)        };      }    });    const onKeypress = (event) => {      const ENTER_CODE = 13;      if (event.keyCode === ENTER_CODE) {        const submitOnEnter = form && form.props.submitOnEnter;        if (!submitOnEnter && props.type !== "textarea") {          preventDefault(event);        }        if (props.type === "search") {          blur();        }      }      emit("keypress", event);    };    const getInputId = () => props.id || `${id}-input`;    const getValidationStatus = () => state.status;    const renderInput = () => {      const controlClass = bem("control", [getProp("inputAlign"), {        error: showError.value,        custom: !!slots.input,        "min-height": props.type === "textarea" && !props.autosize      }]);      if (slots.input) {        return _createVNode("div", {          "class": controlClass,          "onClick": onClickInput        }, [slots.input()]);      }      const inputAttrs = {        id: getInputId(),        ref: inputRef,        name: props.name,        rows: props.rows !== void 0 ? +props.rows : void 0,        class: controlClass,        disabled: getProp("disabled"),        readonly: getProp("readonly"),        autofocus: props.autofocus,        placeholder: props.placeholder,        autocomplete: props.autocomplete,        enterkeyhint: props.enterkeyhint,        "aria-labelledby": props.label ? `${id}-label` : void 0,        onBlur,        onFocus,        onInput,        onClick: onClickInput,        onChange: endComposing,        onKeypress,        onCompositionend: endComposing,        onCompositionstart: startComposing      };      if (props.type === "textarea") {        return _createVNode("textarea", inputAttrs, null);      }      return _createVNode("input", _mergeProps(mapInputType(props.type), inputAttrs), null);    };    const renderLeftIcon = () => {      const leftIconSlot = slots["left-icon"];      if (props.leftIcon || leftIconSlot) {        return _createVNode("div", {          "class": bem("left-icon"),          "onClick": onClickLeftIcon        }, [leftIconSlot ? leftIconSlot() : _createVNode(Icon, {          "name": props.leftIcon,          "classPrefix": props.iconPrefix        }, null)]);      }    };    const renderRightIcon = () => {      const rightIconSlot = slots["right-icon"];      if (props.rightIcon || rightIconSlot) {        return _createVNode("div", {          "class": bem("right-icon"),          "onClick": onClickRightIcon        }, [rightIconSlot ? rightIconSlot() : _createVNode(Icon, {          "name": props.rightIcon,          "classPrefix": props.iconPrefix        }, null)]);      }    };    const renderWordLimit = () => {      if (props.showWordLimit && props.maxlength) {        const count = getStringLength(getModelValue());        return _createVNode("div", {          "class": bem("word-limit")        }, [_createVNode("span", {          "class": bem("word-num")        }, [count]), _createTextVNode("/"), props.maxlength]);      }    };    const renderMessage = () => {      if (form && form.props.showErrorMessage === false) {        return;      }      const message = props.errorMessage || state.validateMessage;      if (message) {        const slot = slots["error-message"];        const errorMessageAlign = getProp("errorMessageAlign");        return _createVNode("div", {          "class": bem("error-message", errorMessageAlign)        }, [slot ? slot({          message        }) : message]);      }    };    const renderLabel = () => {      const colon = getProp("colon") ? ":" : "";      if (slots.label) {        return [slots.label(), colon];      }      if (props.label) {        return _createVNode("label", {          "id": `${id}-label`,          "for": getInputId()        }, [props.label + colon]);      }    };    const renderFieldBody = () => [_createVNode("div", {      "class": bem("body")    }, [renderInput(), showClear.value && _createVNode(Icon, {      "ref": clearIconRef,      "name": props.clearIcon,      "class": bem("clear")    }, null), renderRightIcon(), slots.button && _createVNode("div", {      "class": bem("button")    }, [slots.button()])]), renderWordLimit(), renderMessage()];    useExpose({      blur,      focus,      validate,      formValue,      resetValidation,      getValidationStatus    });    provide(CUSTOM_FIELD_INJECTION_KEY, {      customValue,      resetValidation,      validateWithTrigger    });    watch(() => props.modelValue, () => {      updateValue(getModelValue());      resetValidation();      validateWithTrigger("onChange");      nextTick(adjustTextareaSize);    });    onMounted(() => {      updateValue(getModelValue(), props.formatTrigger);      nextTick(adjustTextareaSize);    });    useEventListener("touchstart", onClear, {      target: computed(() => {        var _a;        return (_a = clearIconRef.value) == null ? void 0 : _a.$el;      })    });    return () => {      const disabled = getProp("disabled");      const labelAlign = getProp("labelAlign");      const Label = renderLabel();      const LeftIcon = renderLeftIcon();      return _createVNode(Cell, {        "size": props.size,        "icon": props.leftIcon,        "class": bem({          error: showError.value,          disabled,          [`label-${labelAlign}`]: labelAlign        }),        "center": props.center,        "border": props.border,        "isLink": props.isLink,        "clickable": props.clickable,        "titleStyle": labelStyle.value,        "valueClass": bem("value"),        "titleClass": [bem("label", [labelAlign, {          required: props.required        }]), props.labelClass],        "arrowDirection": props.arrowDirection      }, {        icon: LeftIcon ? () => LeftIcon : null,        title: Label ? () => Label : null,        value: renderFieldBody,        extra: slots.extra      });    };  }});export {  stdin_default as default,  fieldSharedProps};
 |