2019-10-07 10:24:30 -04:00
|
|
|
<template>
|
|
|
|
<div
|
|
|
|
class="formulate-input"
|
|
|
|
:data-classification="classification"
|
2019-11-06 17:17:19 -05:00
|
|
|
:data-has-errors="hasErrors"
|
2020-03-06 16:10:25 -05:00
|
|
|
:data-is-showing-errors="hasVisibleErrors"
|
2019-10-07 10:24:30 -04:00
|
|
|
:data-type="type"
|
|
|
|
>
|
|
|
|
<div class="formulate-input-wrapper">
|
|
|
|
<slot
|
2020-01-28 12:12:08 -05:00
|
|
|
v-if="context.hasLabel && context.labelPosition === 'before'"
|
2019-10-07 10:24:30 -04:00
|
|
|
name="label"
|
|
|
|
v-bind="context"
|
|
|
|
>
|
|
|
|
<label
|
|
|
|
class="formulate-input-label formulate-input-label--before"
|
2019-10-08 13:50:53 -04:00
|
|
|
:for="context.attributes.id"
|
2019-10-07 10:24:30 -04:00
|
|
|
v-text="context.label"
|
|
|
|
/>
|
|
|
|
</slot>
|
2020-01-28 12:12:08 -05:00
|
|
|
<slot
|
|
|
|
name="element"
|
|
|
|
v-bind="context"
|
|
|
|
>
|
2019-10-07 10:24:30 -04:00
|
|
|
<component
|
|
|
|
:is="context.component"
|
|
|
|
:context="context"
|
2020-01-28 12:12:08 -05:00
|
|
|
>
|
|
|
|
<slot v-bind="context" />
|
|
|
|
</component>
|
2019-10-07 10:24:30 -04:00
|
|
|
</slot>
|
|
|
|
<slot
|
2020-01-28 12:12:08 -05:00
|
|
|
v-if="context.hasLabel && context.labelPosition === 'after'"
|
2019-10-07 10:24:30 -04:00
|
|
|
name="label"
|
|
|
|
v-bind="context.label"
|
|
|
|
>
|
|
|
|
<label
|
|
|
|
class="formulate-input-label formulate-input-label--after"
|
2019-10-08 13:50:53 -04:00
|
|
|
:for="context.attributes.id"
|
2019-10-07 10:24:30 -04:00
|
|
|
v-text="context.label"
|
|
|
|
/>
|
|
|
|
</slot>
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
v-if="help"
|
|
|
|
class="formulate-input-help"
|
|
|
|
v-text="help"
|
|
|
|
/>
|
2020-03-06 16:10:25 -05:00
|
|
|
<FormulateErrors
|
|
|
|
v-if="!disableErrors"
|
|
|
|
:type="`input`"
|
|
|
|
:errors="explicitErrors"
|
|
|
|
:field-name="nameOrFallback"
|
|
|
|
:validation-errors="validationErrors"
|
|
|
|
:show-validation-errors="showValidationErrors"
|
2019-11-06 17:17:19 -05:00
|
|
|
/>
|
2019-10-07 10:24:30 -04:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import context from './libs/context'
|
2020-03-04 13:45:37 -05:00
|
|
|
import { shallowEqualObjects, parseRules, snakeToCamel } from './libs/utils'
|
2020-02-28 11:19:44 -05:00
|
|
|
import nanoid from 'nanoid/non-secure'
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
|
|
export default {
|
|
|
|
name: 'FormulateInput',
|
|
|
|
inheritAttrs: false,
|
2019-10-08 23:54:16 -04:00
|
|
|
inject: {
|
|
|
|
formulateFormSetter: { default: undefined },
|
2019-11-13 16:10:17 -05:00
|
|
|
formulateFormRegister: { default: undefined },
|
|
|
|
getFormValues: { default: () => () => ({}) }
|
2019-10-08 23:54:16 -04:00
|
|
|
},
|
2019-10-07 10:24:30 -04:00
|
|
|
model: {
|
|
|
|
prop: 'formulateValue',
|
|
|
|
event: 'input'
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
type: {
|
|
|
|
type: String,
|
|
|
|
default: 'text'
|
|
|
|
},
|
2019-10-08 23:54:16 -04:00
|
|
|
name: {
|
2020-01-28 16:53:13 -05:00
|
|
|
type: [String, Boolean],
|
2019-10-08 23:54:16 -04:00
|
|
|
default: true
|
|
|
|
},
|
|
|
|
/* eslint-disable */
|
2019-10-07 10:24:30 -04:00
|
|
|
formulateValue: {
|
2019-10-29 23:33:31 -04:00
|
|
|
default: ''
|
2019-10-07 10:24:30 -04:00
|
|
|
},
|
|
|
|
value: {
|
|
|
|
default: false
|
|
|
|
},
|
2019-10-08 23:54:16 -04:00
|
|
|
/* eslint-enable */
|
2019-10-07 10:24:30 -04:00
|
|
|
options: {
|
|
|
|
type: [Object, Array, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
optionGroups: {
|
|
|
|
type: [Object, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
id: {
|
|
|
|
type: [String, Boolean, Number],
|
2019-10-08 13:50:53 -04:00
|
|
|
default: false
|
2019-10-07 10:24:30 -04:00
|
|
|
},
|
|
|
|
label: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
labelPosition: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
help: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
2019-10-08 13:50:53 -04:00
|
|
|
},
|
|
|
|
debug: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false
|
2019-11-06 17:17:19 -05:00
|
|
|
},
|
|
|
|
errors: {
|
|
|
|
type: [String, Array, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
validation: {
|
|
|
|
type: [String, Boolean, Array],
|
|
|
|
default: false
|
|
|
|
},
|
2019-11-13 16:10:17 -05:00
|
|
|
validationName: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
error: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
errorBehavior: {
|
2019-11-06 17:17:19 -05:00
|
|
|
type: String,
|
|
|
|
default: 'blur',
|
|
|
|
validator: function (value) {
|
|
|
|
return ['blur', 'live'].includes(value)
|
|
|
|
}
|
|
|
|
},
|
2019-11-13 16:10:17 -05:00
|
|
|
showErrors: {
|
|
|
|
type: Boolean,
|
2019-11-06 17:17:19 -05:00
|
|
|
default: false
|
2019-11-15 14:44:01 -05:00
|
|
|
},
|
2019-11-21 00:29:28 -05:00
|
|
|
imageBehavior: {
|
|
|
|
type: String,
|
|
|
|
default: 'preview'
|
2019-11-15 14:44:01 -05:00
|
|
|
},
|
|
|
|
uploadUrl: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
uploader: {
|
|
|
|
type: [Function, Object, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
2019-11-19 07:15:13 -05:00
|
|
|
uploadBehavior: {
|
2019-11-20 23:16:31 -05:00
|
|
|
type: String,
|
|
|
|
default: 'live'
|
2019-11-19 07:15:13 -05:00
|
|
|
},
|
|
|
|
preventWindowDrops: {
|
2019-11-15 14:44:01 -05:00
|
|
|
type: Boolean,
|
|
|
|
default: true
|
2020-02-27 01:18:51 -05:00
|
|
|
},
|
|
|
|
showValue: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
validationMessages: {
|
|
|
|
type: Object,
|
|
|
|
default: () => ({})
|
|
|
|
},
|
|
|
|
validationRules: {
|
|
|
|
type: Object,
|
|
|
|
default: () => ({})
|
2020-03-01 22:29:54 -05:00
|
|
|
},
|
|
|
|
checked: {
|
|
|
|
type: [String, Boolean],
|
|
|
|
default: false
|
2020-03-06 16:10:25 -05:00
|
|
|
},
|
|
|
|
disableErrors: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false
|
2019-10-08 13:50:53 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
data () {
|
|
|
|
return {
|
|
|
|
defaultId: nanoid(9),
|
2019-10-29 23:33:31 -04:00
|
|
|
localAttributes: {},
|
2020-03-01 22:29:54 -05:00
|
|
|
internalModelProxy: this.getInitialValue(),
|
2019-11-13 16:10:17 -05:00
|
|
|
behavioralErrorVisibility: (this.errorBehavior === 'live'),
|
2020-01-28 16:53:13 -05:00
|
|
|
formShouldShowErrors: false,
|
2019-11-20 23:16:31 -05:00
|
|
|
validationErrors: [],
|
|
|
|
pendingValidation: Promise.resolve()
|
2019-10-07 10:24:30 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
2019-10-08 13:50:53 -04:00
|
|
|
...context,
|
2019-10-07 10:24:30 -04:00
|
|
|
classification () {
|
|
|
|
const classification = this.$formulate.classify(this.type)
|
|
|
|
return (classification === 'box' && this.options) ? 'group' : classification
|
|
|
|
},
|
|
|
|
component () {
|
2019-10-08 13:50:53 -04:00
|
|
|
return (this.classification === 'group') ? 'FormulateInputGroup' : this.$formulate.component(this.type)
|
2020-03-04 13:45:37 -05:00
|
|
|
},
|
|
|
|
parsedValidationRules () {
|
|
|
|
const parsedValidationRules = {}
|
|
|
|
Object.keys(this.validationRules).forEach((key) => {
|
|
|
|
parsedValidationRules[snakeToCamel(key)] = this.validationRules[key]
|
|
|
|
})
|
|
|
|
return parsedValidationRules
|
|
|
|
},
|
|
|
|
messages () {
|
|
|
|
const messages = {}
|
|
|
|
Object.keys(this.validationMessages).forEach((key) => {
|
|
|
|
messages[snakeToCamel(key)] = this.validationMessages[key]
|
|
|
|
})
|
|
|
|
return messages
|
2019-10-08 13:50:53 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
watch: {
|
|
|
|
'$attrs': {
|
|
|
|
handler (value) {
|
|
|
|
this.updateLocalAttributes(value)
|
|
|
|
},
|
|
|
|
deep: true
|
2019-10-29 23:33:31 -04:00
|
|
|
},
|
|
|
|
internalModelProxy (newValue, oldValue) {
|
2019-11-07 17:03:34 -05:00
|
|
|
this.performValidation()
|
2019-10-29 23:33:31 -04:00
|
|
|
if (!this.isVmodeled && !shallowEqualObjects(newValue, oldValue)) {
|
|
|
|
this.context.model = newValue
|
|
|
|
}
|
|
|
|
},
|
|
|
|
formulateValue (newValue, oldValue) {
|
|
|
|
if (this.isVmodeled && !shallowEqualObjects(newValue, oldValue)) {
|
|
|
|
this.context.model = newValue
|
|
|
|
}
|
2019-10-08 13:50:53 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
created () {
|
2020-03-01 22:29:54 -05:00
|
|
|
this.applyInitialValue()
|
2019-10-08 23:54:16 -04:00
|
|
|
if (this.formulateFormRegister && typeof this.formulateFormRegister === 'function') {
|
2019-10-29 23:33:31 -04:00
|
|
|
this.formulateFormRegister(this.nameOrFallback, this)
|
2019-10-08 23:54:16 -04:00
|
|
|
}
|
2019-10-08 13:50:53 -04:00
|
|
|
this.updateLocalAttributes(this.$attrs)
|
2019-11-07 17:03:34 -05:00
|
|
|
this.performValidation()
|
2019-10-08 13:50:53 -04:00
|
|
|
},
|
|
|
|
methods: {
|
2020-03-01 22:29:54 -05:00
|
|
|
getInitialValue () {
|
|
|
|
// Manually request classification, pre-computed props
|
|
|
|
var classification = this.$formulate.classify(this.type)
|
|
|
|
classification = (classification === 'box' && this.options) ? 'group' : classification
|
|
|
|
if (classification === 'box' && this.checked) {
|
|
|
|
return this.value || true
|
|
|
|
} else if (Object.prototype.hasOwnProperty.call(this.$options.propsData, 'value') && classification !== 'box') {
|
|
|
|
return this.value
|
|
|
|
} else if (Object.prototype.hasOwnProperty.call(this.$options.propsData, 'formulateValue')) {
|
|
|
|
return this.formulateValue
|
|
|
|
}
|
|
|
|
return ''
|
|
|
|
},
|
|
|
|
applyInitialValue () {
|
|
|
|
// This should only be run immediately on created and ensures that the
|
|
|
|
// proxy and the model are both the same before any additional registration.
|
|
|
|
if (
|
|
|
|
!shallowEqualObjects(this.context.model, this.internalModelProxy) &&
|
|
|
|
// we dont' want to set the model if we are a sub-box of a multi-box field
|
|
|
|
(Object.prototype.hasOwnProperty(this.$options.propsData, 'options') && this.classification === 'box')
|
|
|
|
) {
|
|
|
|
this.context.model = this.internalModelProxy
|
|
|
|
}
|
|
|
|
},
|
2019-10-08 13:50:53 -04:00
|
|
|
updateLocalAttributes (value) {
|
|
|
|
if (!shallowEqualObjects(value, this.localAttributes)) {
|
|
|
|
this.localAttributes = value
|
|
|
|
}
|
2019-11-07 17:03:34 -05:00
|
|
|
},
|
|
|
|
performValidation () {
|
2020-03-04 13:45:37 -05:00
|
|
|
const rules = parseRules(this.validation, this.$formulate.rules(this.parsedValidationRules))
|
2019-11-20 23:16:31 -05:00
|
|
|
this.pendingValidation = Promise.all(
|
2019-11-07 17:03:34 -05:00
|
|
|
rules.map(([rule, args]) => {
|
2020-02-27 01:18:51 -05:00
|
|
|
var res = rule({
|
2020-02-25 17:32:40 -05:00
|
|
|
value: this.context.model,
|
|
|
|
getFormValues: this.getFormValues.bind(this),
|
|
|
|
name: this.context.name
|
|
|
|
}, ...args)
|
2020-02-27 01:18:51 -05:00
|
|
|
res = (res instanceof Promise) ? res : Promise.resolve(res)
|
|
|
|
return res.then(res => res ? false : this.getValidationMessage(rule, args))
|
2019-11-07 17:03:34 -05:00
|
|
|
})
|
|
|
|
)
|
|
|
|
.then(result => result.filter(result => result))
|
2019-11-21 00:29:28 -05:00
|
|
|
.then(errorMessages => { this.validationErrors = errorMessages })
|
2019-11-20 23:16:31 -05:00
|
|
|
return this.pendingValidation
|
|
|
|
},
|
2020-02-27 01:18:51 -05:00
|
|
|
getValidationMessage (rule, args) {
|
|
|
|
return this.getValidationFunction(rule)({
|
|
|
|
args,
|
|
|
|
name: this.mergedValidationName,
|
|
|
|
value: this.context.model,
|
|
|
|
vm: this,
|
|
|
|
formValues: this.getFormValues()
|
|
|
|
})
|
|
|
|
},
|
|
|
|
getValidationFunction (rule) {
|
2020-03-04 13:45:37 -05:00
|
|
|
let ruleName = rule.name.substr(0, 1) === '_' ? rule.name.substr(1) : rule.name
|
|
|
|
ruleName = snakeToCamel(ruleName)
|
|
|
|
if (this.messages && typeof this.messages === 'object' && typeof this.messages[ruleName] !== 'undefined') {
|
|
|
|
switch (typeof this.messages[ruleName]) {
|
2020-02-27 01:18:51 -05:00
|
|
|
case 'function':
|
2020-03-04 13:45:37 -05:00
|
|
|
return this.messages[ruleName]
|
2020-02-27 01:18:51 -05:00
|
|
|
case 'string':
|
2020-03-04 13:45:37 -05:00
|
|
|
return () => this.messages[ruleName]
|
2020-02-27 01:18:51 -05:00
|
|
|
}
|
|
|
|
}
|
2020-03-07 09:03:59 -05:00
|
|
|
return (context) => this.$formulate.validationMessage(rule.name, context, this)
|
2020-02-27 01:18:51 -05:00
|
|
|
},
|
2019-11-20 23:16:31 -05:00
|
|
|
hasValidationErrors () {
|
|
|
|
return new Promise(resolve => {
|
|
|
|
this.$nextTick(() => {
|
|
|
|
this.pendingValidation.then(() => resolve(!!this.validationErrors.length))
|
|
|
|
})
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|