vxe-pc-ui
Version:
A vue based PC component library
174 lines (150 loc) • 6.13 kB
text/typescript
import { h, computed, reactive, inject, PropType } from 'vue'
import { defineVxeComponent } from '../../ui/src/comp'
import XEUtils from 'xe-utils'
import { getFuncText } from '../../ui/src/utils'
import { getConfig, createEvent, useSize } from '../../ui'
import type { VxeRadioButtonPropTypes, VxeRadioGroupConstructor, RadioButtonReactData, RadioButtonPrivateMethods, VxeRadioButtonConstructor, VxeRadioButtonEmits, VxeRadioGroupPrivateMethods, RadioButtonMethods, VxeFormConstructor, VxeFormPrivateMethods, VxeFormDefines, ValueOf } from '../../../types'
export default defineVxeComponent({
name: 'VxeRadioButton',
props: {
modelValue: [String, Number, Boolean] as PropType<VxeRadioButtonPropTypes.ModelValue>,
checkedValue: {
type: [String, Number, Boolean] as PropType<VxeRadioButtonPropTypes.CheckedValue>,
default: undefined
},
title: [String, Number] as PropType<VxeRadioButtonPropTypes.Title>,
content: [String, Number] as PropType<VxeRadioButtonPropTypes.Content>,
disabled: {
type: Boolean as PropType<VxeRadioButtonPropTypes.Disabled>,
default: null
},
strict: {
type: Boolean as PropType<VxeRadioButtonPropTypes.Strict>,
default: () => getConfig().radioButton.strict
},
size: {
type: String as PropType<VxeRadioButtonPropTypes.Size>,
default: () => getConfig().radioButton.size || getConfig().size
},
/**
* 已废弃,被 checkedValue 替换
*/
label: {
type: [String, Number, Boolean] as PropType<VxeRadioButtonPropTypes.Label>,
default: null
}
},
emits: [
'update:modelValue',
'change'
] as VxeRadioButtonEmits,
setup (props, context) {
const { slots, emit } = context
const $xeForm = inject<VxeFormConstructor & VxeFormPrivateMethods | null>('$xeForm', null)
const formItemInfo = inject<VxeFormDefines.ProvideItemInfo | null>('xeFormItemInfo', null)
const $xeRadioGroup = inject<(VxeRadioGroupConstructor & VxeRadioGroupPrivateMethods) | null>('$xeRadioGroup', null)
const xID = XEUtils.uniqueId()
const reactData = reactive<RadioButtonReactData>({
})
const { computeSize } = useSize(props)
const $xeRadioButton = {
xID,
props,
context,
reactData
} as unknown as VxeRadioButtonConstructor
const computeIsDisabled = computed(() => {
const { disabled } = props
if (disabled === null) {
if ($xeRadioGroup) {
const { computeIsDisabled } = $xeRadioGroup.getComputeMaps()
return computeIsDisabled.value
}
}
return disabled
})
const computeName = computed(() => {
return $xeRadioGroup ? $xeRadioGroup.name : null
})
const computeStrict = computed(() => {
return $xeRadioGroup ? $xeRadioGroup.props.strict : props.strict
})
const computeChecked = computed(() => {
const { label, checkedValue } = props
const radioValue = XEUtils.isUndefined(checkedValue) ? label : checkedValue
return $xeRadioGroup ? $xeRadioGroup.props.modelValue === radioValue : props.modelValue === radioValue
})
const dispatchEvent = (type: ValueOf<VxeRadioButtonEmits>, params: Record<string, any>, evnt: Event | null) => {
emit(type, createEvent(evnt, { $radioGroup: $xeRadioGroup }, params))
}
const radioButtonMethods: RadioButtonMethods = {
dispatchEvent
}
const radioButtonPrivateMethods: RadioButtonPrivateMethods = {
}
Object.assign($xeRadioButton, radioButtonMethods, radioButtonPrivateMethods)
const handleValue = (checkedValue: VxeRadioButtonPropTypes.CheckedValue, evnt: Event) => {
const { content } = props
if ($xeRadioGroup) {
$xeRadioGroup.handleChecked({ label: checkedValue, checkedValue, checkedLabel: content }, evnt)
} else {
emit('update:modelValue', checkedValue)
dispatchEvent('change', { value: checkedValue, label: checkedValue, checkedValue, checkedLabel: content }, evnt)
// 自动更新校验状态
if ($xeForm && formItemInfo) {
$xeForm.triggerItemEvent(evnt, formItemInfo.itemConfig.field, checkedValue)
}
}
}
const changeEvent = (evnt: Event) => {
const isDisabled = computeIsDisabled.value
if (!isDisabled) {
const { label, checkedValue } = props
const radioValue = XEUtils.isUndefined(checkedValue) ? label : checkedValue
handleValue(radioValue, evnt)
}
}
const clickEvent = (evnt: Event) => {
const isDisabled = computeIsDisabled.value
const isStrict = computeStrict.value
if (!isDisabled && !isStrict) {
const { label, checkedValue } = props
const radioValue = XEUtils.isUndefined(checkedValue) ? label : checkedValue
if (radioValue === ($xeRadioGroup ? $xeRadioGroup.props.modelValue : props.modelValue)) {
handleValue(null, evnt)
}
}
}
const renderVN = () => {
const { label, checkedValue } = props
const radioValue = XEUtils.isUndefined(checkedValue) ? label : checkedValue
const vSize = computeSize.value
const isDisabled = computeIsDisabled.value
const name = computeName.value
const isChecked = computeChecked.value
return h('label', {
key: radioValue,
class: ['vxe-radio vxe-radio--button', {
[`size--${vSize}`]: vSize,
'is--disabled': isDisabled
}],
title: props.title
}, [
h('input', {
class: 'vxe-radio--input',
type: 'radio',
name,
checked: isChecked,
disabled: isDisabled,
onChange: changeEvent,
onClick: clickEvent
}),
h('span', {
class: 'vxe-radio--label'
}, slots.default ? slots.default({}) : getFuncText(props.content))
])
}
$xeRadioButton.renderVN = renderVN
return renderVN
}
})