拆分数据

master
于肖磊 2025-07-16 11:57:25 +08:00
parent a38ae503ff
commit ed7e48d449
5 changed files with 1588 additions and 0 deletions

View File

@ -0,0 +1,98 @@
<template>
<view v-if="form.seckill_subscript_show == '1'" class="corner-marker" :style="corner_marker">
<view class="flex-row nowrap" :style="corner_img_marker">
<template v-if="form.subscript_type == 'img-icon'">
<template v-if="!isEmpty(form.subscript_img_src)">
<image :src="form.subscript_img_src[0].url" mode="aspectFill" :style="img_style" />
</template>
<template v-else>
<iconfont :name="'icon-' + form.subscript_icon_class" propContainerDisplay="flex" :size="new_style.subscript_style.text_or_icon_size * 2 + 'rpx'" :color="new_style.subscript_style.text_or_icon_color"></iconfont>
</template>
</template>
<template v-else>
<span class="text-line-1" :style="text_size">{{ form.subscript_text }}</span>
</template>
</view>
</view>
</template>
<script>
import { common_img_computer, common_styles_computer, isEmpty } from '@/common/js/common/common.js';
import iconfont from '@/components/iconfont/iconfont';
export default {
components: {
iconfont,
},
props: {
propValue: {
type: Object,
default: () => ({}),
},
propType: {
type: String,
default: 'outer',
},
},
data() {
return {
form: {},
new_style: {},
corner_img_marker: '',
img_style: '',
text_size: '',
};
},
created() {
this.init();
},
methods: {
isEmpty,
//
init() {
if (!isEmpty(this.propValue)) {
const new_content = this.propValue.content || {};
const new_style = this.propType == 'outer' ? this.propValue.style || {} : { subscript_style: this.propValue.style || {} };
if (!isEmpty(new_style.subscript_style)) {
//
this.setData({
form: new_content,
new_style: new_style,
corner_marker: this.get_corner_marker(new_style),
text_size: `font-size: ${ new_style.subscript_style.text_or_icon_size * 2 }rpx;color: ${ new_style.subscript_style.text_or_icon_color };`,
corner_img_marker: common_img_computer(new_style.subscript_style),
img_style: `height: ${new_style.subscript_style.img_height * 2}rpx; width: ${new_style.subscript_style.img_width * 2}rpx`,
});
}
}
},
get_corner_marker(new_style) {
const { subscript_style } = new_style;
let location = common_styles_computer(subscript_style);
//
const { seckill_subscript_location, top_or_bottom_spacing, left_or_right_spacing } = subscript_style;
// 线
if (seckill_subscript_location == 'top-left') {
location += `top: ${ top_or_bottom_spacing * 2 }rpx;left: ${ left_or_right_spacing * 2 }rpx;`;
} else if (seckill_subscript_location == 'top-center') {
location += 'top: 0;left: 50%;transform: translateX(-50%);';
} else if (seckill_subscript_location == 'top-right') {
location += `top: ${ top_or_bottom_spacing * 2 }rpx;right:${ left_or_right_spacing * 2 }rpx;`;
} else if (seckill_subscript_location == 'bottom-left') {
location += `bottom: ${ top_or_bottom_spacing * 2 }rpx;left: ${ left_or_right_spacing * 2 }rpx;`;
} else if (seckill_subscript_location == 'bottom-center') {
location += 'bottom: 0;left: 50%;transform: translateX(-50%);';
} else if (seckill_subscript_location == 'bottom-right') {
location += `bottom: ${ top_or_bottom_spacing * 2 }rpx;right: ${ left_or_right_spacing * 2 }rpx;`;
}
return location;
},
},
};
</script>
<style>
.corner-marker {
position: absolute;
max-width: 100%;
}
</style>

View File

@ -0,0 +1,714 @@
<template>
<view class="wh-auto ht-auto pr">
<view v-for="(item, index) in filteredDiyData" :key="index" :class="(is_custom ? 'pa ' : (flex_direction == 'row' ? 'row-item ' : 'column-item mb-10 ')) + (item.com_data.common_config.is_error == '1' ? ' item_error' : '')" :data-id="item.id" :data-location-x="item.location.x" :data-location-y="item.location.y" :style="(item.key == 'auxiliary-line' ? 'border-bottom: 0rpx; ' : '') + (is_custom ? ('left:' + item.location.x + 'px;top:' + item.location.y + 'px;width:' + item.com_data.com_width + 'px;height:' + item.com_data.com_height + 'px;z-index:' + (z_index_id == item.id ? '999999' : (item.is_enable == '1' ? (data_list.length - 1 > 0 ? (data_list.length - 1) - index : 0) : -999)) + ';') : '')">
<view :class="'wh-auto ht-auto ' + (flex_direction == 'row' ? (['video', 'img', 'upload-img', 'upload-video', 'multi-text'].includes(item.key) ? 'flex-row align-s gap-10' : 'flex-row align-b gap-10') : 'flex-col gap-10')">
<view v-if="(!is_custom && !['auxiliary-line', 'subform'].includes(item.key)) || (is_custom && !['img', 'video', 'auxiliary-line', 'rect', 'round', 'subform'].includes(item.key))" class="field-label flex-row align-c gap-10" :style="field_label_style + (flex_direction == 'row' && ['upload-img', 'upload-video'].includes(item.key) ? 'padding-top: 12rpx;line-height: 120rpx;' : '') + (flex_direction == 'row' && ['multi-text'].includes(item.key) ? 'padding-top: 18rpx;' : '')">
<view class="flex-row align-c" :style="title_style">{{ item.com_data.title }}<view v-if="item.com_data.is_required == '1'" class="required">*</view></view>
<view v-if="item.com_data.common_config.help_is_show == '1' && !isEmpty(item.com_data.common_config.help_explain)" :data-value="item.com_data.common_config.help_explain" @tap="help_icon_event">
<iconfont name="icon-miaosha-hdgz" :size="help_icon_style" color="#999"></iconfont>
</view>
</view>
<view class="flex-1 wh-auto ht-auto flex-col gap-5 oh">
<!-- 输入框 -->
<view v-if="['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'single-text'" :style="item.com_data.common_style">
<component-input :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change"></component-input>
</view>
<!-- 多行输入框 -->
<view v-else-if="item.key == 'multi-text'" :style="item.com_data.common_style + 'padding: 18rpx 22rpx;'">
<component-textarea :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change"></component-textarea>
</view>
<!-- 复选按钮组 -->
<view v-else-if="['select-multi', 'checkbox'].includes(item.key) && item.com_data.type == 'checkbox' && flex_direction !== 'row'">
<component-checkbox :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change" @dataOptionChange="data_option_change"></component-checkbox>
</view>
<!-- 单选按钮组 -->
<view v-else-if="['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'radio-btns' && flex_direction !== 'row'">
<component-radio :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change"></component-radio>
</view>
<!-- 下拉框 -->
<view v-else-if="(['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'select') || (['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'radio-btns' && flex_direction == 'row')" :style="item.com_data.common_style">
<component-select :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propDirection="flex_direction" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-select>
</view>
<!-- 下拉复选框 -->
<view v-else-if="(['select-multi', 'checkbox'].includes(item.key) && item.com_data.type == 'select-multi') || (['select-multi', 'checkbox'].includes(item.key) && item.com_data.type == 'checkbox' && flex_direction == 'row')" :style="item.com_data.common_style">
<component-select-multi :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propDirection="flex_direction" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change" @dataOptionChange="data_option_change" @zIndexChange="z_index_change"></component-select-multi>
</view>
<!-- 数字 -->
<view v-else-if="item.key == 'number'" :style="item.com_data.common_style">
<component-number :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change"></component-number>
</view>
<!-- 时间选择器 -->
<view v-else-if="item.key == 'date'" :style="item.com_data.common_style">
<component-date :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-date>
</view>
<!-- 时间选择器组 -->
<view v-else-if="item.key == 'date-group'" :style="item.com_data.common_style">
<component-date-group :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-date-group>
</view>
<!-- 地址 -->
<view v-else-if="item.key == 'address'">
<component-address :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataCheck="data_check" @dataChange="data_change" @regionEvent="region_event" @dataAddressChange="data_address_change" @zIndexChange="z_index_change"></component-address>
</view>
<!-- 手机 -->
<view v-else-if="item.key == 'phone'">
<component-phone :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataCheck="data_check" @dataChange="data_change" @dataCodeCheck="data_code_check" @dataCodeChage="data_code_change" @zIndexChange="z_index_change"></component-phone>
</view>
<!-- 密码 -->
<view v-else-if="item.key == 'pwd'" :style="item.com_data.common_style">
<component-pwd :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataCheck="data_check" @dataChange="data_change"></component-pwd>
</view>
<!-- 评分 -->
<view v-else-if="item.key == 'score'">
<component-score :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataCheck="data_check" @dataChange="data_change"></component-score>
</view>
<!-- 图片 -->
<view v-else-if="item.key == 'img'" :class="is_custom ? 'wh-auto ht-auto' : ''">
<component-image :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" :propIsCustom="is_custom"></component-image>
</view>
<!-- 视频 -->
<view v-else-if="item.key == 'video'" :class="is_custom ? 'wh-auto ht-auto' : ''">
<component-video :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" :propIsCustom="is_custom"></component-video>
</view>
<!-- 文本 -->
<view v-else-if="item.key == 'text'">
<component-text :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction"></component-text>
</view>
<!-- 文件 -->
<view v-else-if="item.key == 'attachments'">
<component-attachments :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction"></component-attachments>
</view>
<!-- 辅助线 -->
<view v-else-if="item.key == 'auxiliary-line'">
<component-auxiliary-line :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction"></component-auxiliary-line>
</view>
<!-- 上传视频或图片 -->
<view v-else-if="['upload-img', 'upload-video'].includes(item.key)" :class="flex_direction == 'row' ? 'padding-vertical-sm' : ''">
<component-upload :propValue="item.com_data" :propType="item.key == 'upload-img' ? 'img' : ( item.key == 'upload-video' ? 'video' : 'file')" :propKey="propKey" :propDataFormId="propDataFormId" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataChange="data_change"></component-upload>
</view>
<!-- 定位 -->
<view v-else-if="item.key == 'position'">
<component-position :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataChange="data_change"></component-position>
</view>
<!-- 圆形和矩形 -->
<view v-else-if="['rect', 'round'].includes(item.key)" :class="is_custom ? 'wh-auto ht-auto' : ''">
<component-rect-or-round :propValue="item.com_data" :propKey="propKey"></component-rect-or-round>
</view>
<!-- 子表单 -->
<view v-else-if="item.key == 'subform'">
<component-subform
:propValue="item.com_data"
:propKey="propKey"
:propDataId="item.id"
:propMobile="mobile"
:propIsCustom="is_custom"
:propStyle="component_style"
:propDirection="flex_direction"
:propTitleStyle="title_style"
:propHelpIconStyle="help_icon_style"
:propFieldLabelStyle="field_label_style"
:propDataFormId="propDataFormId"
@helpIconEvent="subform_help_icon_event"
@subformDataChange="subform_data_change"
@zIndexChange="z_index_change"
/>
</view>
<!-- #ifdef H5 || MP-WEIXIN || MP-QQ -->
<!-- 上传文件 -->
<view v-else-if="item.key == 'upload-attachments'">
<component-upload :propValue="item.com_data" propType="file" :propKey="propKey" :propDataFormId="propDataFormId" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataChange="data_change"></component-upload>
</view>
<!-- #endif -->
<!-- #ifdef APP-PLUS || H5 || MP-WEIXIN -->
<!-- 富文本 -->
<view v-else-if="item.key == 'rich-text'" :style="item.com_data.common_style + 'padding:0;'">
<component-rich-text :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="mobile" :propStyle="component_style" :propDirection="flex_direction" @dataChange="data_change"></component-rich-text>
</view>
<!-- #endif -->
<view v-if="!isEmpty(item.com_data.common_config.error_text)" class="field-invalid-info">{{ item.com_data.common_config.error_text }}</view>
</view>
</view>
</view>
<view class="z-i-deep-must">
<uni-popup ref="popup" type="center" border-radius="20rpx">
<view class="popup-content">{{ popup_help_content }}</view>
</uni-popup>
</view>
</view>
</template>
<script>
import { isEmpty, common_form_styles_computer, get_format_checks } from '@/common/js/common/common.js';
import componentInput from '@/pages/form-input/components/form-input/input.vue';
import componentTextarea from '@/pages/form-input/components/form-input/textarea.vue';
import componentCheckbox from '@/pages/form-input/components/form-input/checkbox.vue';
import componentRadio from '@/pages/form-input/components/form-input/radio.vue';
import componentSelect from '@/pages/form-input/components/form-input/select.vue';
import componentNumber from '@/pages/form-input/components/form-input/number.vue';
import componentDate from '@/pages/form-input/components/form-input/date.vue';
import componentDateGroup from '@/pages/form-input/components/form-input/date-group.vue';
import componentAddress from '@/pages/form-input/components/form-input/address.vue';
import componentSelectMulti from '@/pages/form-input/components/form-input/select-multi.vue';
import componentPhone from '@/pages/form-input/components/form-input/phone.vue';
import componentPwd from '@/pages/form-input/components/form-input/pwd.vue';
import componentScore from '@/pages/form-input/components/form-input/score.vue';
import componentImage from '@/pages/form-input/components/form-input/image.vue';
import componentVideo from '@/pages/form-input/components/form-input/video.vue';
import componentText from '@/pages/form-input/components/form-input/text.vue';
import componentAttachments from '@/pages/form-input/components/form-input/attachments.vue';
import componentAuxiliaryLine from '@/pages/form-input/components/form-input/auxiliary-line.vue';
import componentRichText from '@/pages/form-input/components/form-input/rich-text.vue';
import componentUpload from '@/pages/form-input/components/form-input/upload.vue';
import componentPosition from '@/pages/form-input/components/form-input/position.vue';
import componentRectOrRound from '@/pages/form-input/components/form-input/rect-or-round.vue';
import componentSubform from '@/pages/form-input/components/form-input/subform.vue';
export default {
name: 'formInput',
components: {
componentInput,
componentTextarea,
componentCheckbox,
componentRadio,
componentNumber,
componentSelect,
componentSelectMulti,
componentDate,
componentDateGroup,
componentAddress,
componentPhone,
componentPwd,
componentScore,
componentImage,
componentVideo,
componentText,
componentAttachments,
componentAuxiliaryLine,
componentRichText,
componentUpload,
componentPosition,
componentRectOrRound,
componentSubform
},
props: {
propValue: {
type: Object,
default: () => {},
},
propDataFormId: {
type: [String, Number],
default: '',
},
propKey: {
type: [String, Number],
default: 0,
},
},
data() {
return {
data_list: [],
z_index_id: '',
flex_direction: 'row',
field_label_style: '',
title_style: '',
component_style: '',
help_icon_style: '',
mobile: {},
popup_help_content: '',
is_custom: false
}
},
computed: {
filteredDiyData() {
const componentMap = new Map(this.data_list.map(item => [item.id, item]));
//
const list = this.data_list.filter(item => ['single-text', 'select', 'radio-btns'].includes(item.key) && ['select', 'radio-btns'].includes(item.com_data.type) && item.com_data.show_hidden_list.length > 0);
const list_map = list.map(item => ({ id: item.id, list: item.com_data.show_hidden_list }));
return this.data_list.filter(item => {
//
if (item.is_enable !== '1') return false;
if (list_map.length === 0) return true;
//
const isShownByRule = list_map.some(list_item => {
const targetComponent = componentMap.get(list_item.id);
//
if (!targetComponent) return false;
return list_item.list.some(hidden_item => {
//
if (hidden_item.is_show.includes(item.id)) {
return targetComponent.com_data.form_value.includes(hidden_item.value);
} else {
return true;
}
});
});
return isShownByRule;
});
}
},
watch: {
propValue: {
handler(newVal) {
this.init();
},
deep: true
},
propKey(val) {
this.init();
}
},
mounted() {
this.init();
},
methods: {
isEmpty,
init() {
const data = this.propValue;
//
const overall_config = data.config?.overall_config || {};
const mobile = overall_config?.style_settings?.mobile || {};
let diy_data = data.config.diy_data || [];
// #ifndef H5 || MP-WEIXIN || MP-QQ
// H5 qq
diy_data = diy_data.filter(item => item.key !== 'upload-attachments');
// #endif
// #ifndef APP-PLUS || H5 || MP-WEIXIN
// APP-PLUS || H5 || MP-WEIXIN
diy_data = diy_data.filter(item => item.key !== 'rich-text');
// #endif
diy_data.forEach(item => {
//
item.com_data.common_style = this.get_form_border_style(item.com_data.common_config, mobile.flex_direction || 'row', overall_config.type_value);
//
if (item.key == 'subform') {
const { com_data } = item;
item.com_data.data_list = [];
let data_list = [];
com_data.form_value.forEach(item1 => {
const data = JSON.parse(JSON.stringify(com_data?.children || []));
if (data.length > 0) {
data.forEach(child => {
child.com_data.common_style = this.get_form_border_style(child.com_data.common_config, mobile.arrang == 'direction' ? (emobile.flex_direction || 'row') : 'column', overall_config.type_value);
if (!isEmpty(item1[child.id])) {
child.com_data.form_value = item1[child.id];
}
});
data_list.push({
is_expand: false,
data_list: data
});
}
});
item.com_data.data_list = data_list;
}
});
this.setData({
z_index_id: '',
data_list: diy_data,
mobile: mobile,
flex_direction: mobile.flex_direction || 'row',
title_style: `color:#333;font-size:${ mobile.filed_title_size_type == 'big' ? 44 : mobile.filed_title_size_type == 'middle' ? 32 : 24 }rpx;font-weight:blod;`,
component_style: `height:${mobile.filed_title_size_type == 'big' ? 108 : mobile.filed_title_size_type == 'middle' ? 80 : 60}rpx;font-size:${mobile.filed_title_size_type == 'big' ? 44 : mobile.filed_title_size_type == 'middle' ? 32 : 24}rpx;`,
help_icon_style: `font-size:${mobile.help_icon_size_type == 'big' ? 40 : mobile.help_icon_size_type == 'middle' ? 28 : 24}rpx;`,
field_label_style: `${ mobile.flex_direction == 'column'? 'justify-content:flex-start;' : `width:${ mobile.filed_title_width * 2 }rpx;justify-content: ${ mobile.filed_title_justification };` }`,
is_custom: overall_config.type_value == 'free',
});
},
//
get_form_border_style(item, flex_direction, type_value) {
//
if (type_value == 'default') {
return flex_direction == 'row' ? '' : common_form_styles_computer(item) + 'padding: 0px 22rpx;box-sizing:content-box;';
} else {
//
return common_form_styles_computer(item) + 'padding: 0px 22rpx;box-sizing:content-box;';
}
},
help_icon_event(e) {
this.setData({ popup_help_content: e.currentTarget.dataset.value });
this.$refs.popup.open();
},
subform_help_icon_event(e) {
this.setData({ popup_help_content: e });
this.$refs.popup.open();
},
//
data_change(e) {
const { value, id } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data) {
item.com_data.form_value = value;
}
});
this.setData({ data_list: data });
},
//
data_check(e) {
const { is_error, error_text, value, id } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data && item.com_data.common_config) {
item.com_data.form_value = value;
item.com_data.common_config.is_error = is_error;
item.com_data.common_config.error_text = error_text;
}
});
this.setData({ data_list: data });
},
//
data_option_change(e) {
const { list, value, id } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data) {
item.com_data.form_value = value;
item.com_data.custom_option_list = list;
}
});
this.setData({ data_list: data });
},
//
data_code_change(e) {
const { value, id } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data) {
item.com_data.form_value_code = value;
}
});
this.setData({ data_list: data });
},
data_code_check(e) {
const { is_error, error_text, value, id } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data && item.com_data.common_config) {
item.com_data.form_value_code = value;
item.com_data.common_config.is_error = is_error;
item.com_data.common_config.error_text = error_text;
}
});
this.setData({ data_list: data });
},
data_address_change(e) {
const { value, id } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data) {
item.com_data.detailed_value = value;
}
});
this.setData({ data_list: data });
},
//
subform_data_change(e, id) {
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data) {
item.com_data.data_list = e
}
});
this.setData({ data_list: data });
},
//
region_event(e) {
const { value, id, province_name, city_name, county_name } = e;
// id
const data = [...this.data_list];
data.forEach(item => {
if (item.id == id && item.com_data) {
item.com_data.form_value = value;
item.com_data.province_name = province_name;
item.com_data.city_name = city_name;
item.com_data.county_name = county_name;
}
});
this.setData({ data_list: data });
},
submit_click() {
try {
//
const new_data = this.submit_data_check();
//
const data = new_data.find((item) => item.com_data.common_config.is_error === '1' || (item.key === 'subform' && item.com_data.data_list.some((item1) => item1.data_list.some(list_item => list_item.com_data.common_config.is_error === '1'))));
if (!isEmpty(data)) {
let message = '';
if (data.key === 'subform') {
// error
if (data.com_data.common_config.is_error == '1') {
message = `${data.com_data.title}${data.com_data.common_config.error_text}`;
} else {
//
message = `请检查${data.com_data.title}内的填写`;
}
} else {
message = `${data.com_data.title}${data.com_data.common_config.error_text}`;
}
//
const old_data = [...this.data_list];
const data_list = old_data.map(item => {
const match = new_data.find(el => el.id === item.id);
return { ...item, ...match };
});
this.setData({ data_list: data_list });
this.$emit('submitData', { type: 'error', submit_data: {}, message: message });
} else {
this.submit_data_parameter_handle();
}
} catch (error) {
this.$emit('submitData', { type: 'error', submit_data: {}, message: '数据错误'});
}
},
submit_data_parameter_handle() {
try {
const submit_data = {};
const filter_data = ['video', 'img', 'auxiliary-line', 'position', 'rect', 'round', 'text', 'attachments'];
//
this.filteredDiyData.forEach((item) => {
if (!filter_data.includes(item.key)) {
const name = isEmpty(item.form_name) ? item.id : item.form_name;
const value = isEmpty(item.com_data.form_value) ? '' : item.com_data.form_value;
const com_data = item.com_data;
if (item.key ==='subform') {
const data_list = com_data.data_list;
submit_data[name] = data_list.map((subform_item) => {
const data = {};
//
subform_item.data_list.forEach((item1) => {
// name
const subform_name = isEmpty(item1.form_name) ? item1.id : item1.form_name;
const subform_com_data = item1.com_data;
// value
const subform_value = isEmpty(subform_com_data.form_value) ? '' : subform_com_data.form_value;
if (!filter_data.includes(item1.key)) {
if (item1.key == 'address') {
data[`${ subform_name }_province_id`] = subform_value[0] || '';
data[`${ subform_name }_city_id`] = subform_value[1] || '';
data[`${ subform_name }_county_id`] = subform_value[2] || '';
//
submit_data[`${ subform_name }_province_name`] = subform_com_data.province_name || '';
submit_data[`${ subform_name }_city_name`] = subform_com_data.city_name || ''
submit_data[`${ subform_name }_county_name`] = subform_com_data.county_name || ''
} else if (['checkbox', 'select-multi'].includes(item.key)) {
submit_data[subform_name] = subform_value;
if (subform_com_data.is_add_option == '1') {
submit_data[`${ subform_name }_custom_option_list`] = subform_com_data?.custom_option_list || [];
}
} else {
data[subform_name] = subform_value;
}
}
});
return data;
});
} else if (item.key ==='phone') {
submit_data[`${ name }`] = value || '';
//
if (com_data.is_sms_verification == '1') {
submit_data[`${ name }_verify`] = com_data?.form_value_code || '';
}
} else if (item.key == 'address') {
submit_data[`${ name }_province_id`] = value[0] || '';
submit_data[`${ name }_city_id`] = value[1] || '';
submit_data[`${ name }_county_id`] = value[2] || '';
//
submit_data[`${ name }_province_name`] = com_data.province_name || '';
submit_data[`${ name }_city_name`] = com_data.city_name || ''
submit_data[`${ name }_county_name`] = com_data.county_name || ''
//
if (com_data.address_type == 'detailed') {
submit_data[`${ name }_address`] = com_data?.detailed_value || '';
}
} else if (['select', 'radio-btns'].includes(item.key)) {
submit_data[name] = value;
const value_list = com_data.option_list.filter((item) => item.is_other == '1');
if (value_list.length > 0) {
submit_data[`${ name }_other_value`] = com_data?.other_value || '';
}
} else if (['checkbox', 'select-multi'].includes(item.key)) {
submit_data[name] = value;
if (com_data.is_add_option == '1') {
submit_data[`${ name }_custom_option_list`] = com_data?.custom_option_list || [];
}
} else {
submit_data[name] = value;
}
}
});
this.$emit('submitData', { type: 'success', submit_data: submit_data, message: ''});
} catch (error) {
this.$emit('submitData', { type: 'error', submit_data: {}, message: '数据错误'});
}
},
submit_data_check() {
//
const fieldCheckMap = {
'address': { is_format: false, type: 'address' },
'number': { is_format: true, type: 'number' },
'checkbox': { is_format: true, type: 'checkbox' },
'select-multi': { is_format: true, type: 'checkbox' },
'date': { is_format: false, type: 'time' },
'date-group': { is_format: false, type: 'time' },
'single-text': { is_format: false, type: '' },
'multi-text': { is_format: false, type: '' },
'rich-text': { is_format: false, type: '' },
'radio-btns': { is_format: false, type: 'radio' },
'select': { is_format: false, type: 'select' },
'pwd': { is_format: false, type: '' },
'score': { is_format: false, type: 'score' },
'upload-attachments': { is_format: false, type: 'upload' },
'upload-img': { is_format: false, type: 'upload' },
'upload-video': { is_format: false, type: 'upload' }
};
const data = JSON.parse(JSON.stringify(this.filteredDiyData));
//
data?.forEach((item) => {
const com_data = item.com_data;
//
if (com_data.is_required === '1') {
//
if (item.key === 'phone') {
const { is_error = '0', error_text = '' } = this.handlePhoneValidation(com_data);
com_data.common_config.is_error = is_error;
com_data.common_config.error_text = error_text;
}
//
else if (fieldCheckMap[item.key]) {
const { is_format, type } = fieldCheckMap[item.key];
const { is_error = '0', error_text = '' } = get_format_checks(com_data, com_data.form_value, is_format, type);
com_data.common_config.is_error = is_error;
com_data.common_config.error_text = error_text;
}
};
/**
* 子表单处理逻辑
* 1. 检查子表单中是否有必填项
* 2. 验证子表单整体必填性
* 3. 处理子表单内各字段的验证
*/
if (item.key === 'subform') {
//
if (com_data.is_required === '1' && com_data.data_list.length <= 0) {
com_data.common_config.is_error = '1';
com_data.common_config.error_text = '请填写至少一条记录';
} else {
com_data.common_config.is_error = '0';
com_data.common_config.error_text = '';
}
//
if (com_data.data_list.length > 0) {
//
com_data.data_list.forEach((form_item, index) => {
//
const form_data = this.filtered_Data(form_item.data_list);
form_data.forEach((data_item) => {
//
if (data_item.com_data.is_required !== '1') return;
//
const checkConfig = fieldCheckMap[data_item.key];
if (checkConfig) {
//
if (data_item.key === 'phone') {
const { is_error = '0', error_text = '' } = this.handlePhoneValidation(data_item);
data_item.com_data.common_config.is_error = is_error;
data_item.com_data.common_config.error_text = error_text;
}
//
else if (fieldCheckMap[data_item.key]) {
const { is_format, type } = fieldCheckMap[data_item.key];
const { is_error = '0', error_text = '' } = get_format_checks(data_item.com_data, data_item.com_data.form_value, is_format, type);
data_item.com_data.common_config.is_error = is_error;
data_item.com_data.common_config.error_text = error_text;
}
}
});
});
}
}
});
return data;
},
//
handlePhoneValidation(com_data) {
if (com_data.is_sms_verification === '1' && com_data.is_required === '1' && isEmpty(com_data.form_value_code)) {
com_data.common_config.is_error = '1';
com_data.common_config.error_text = '短信验证码不能为空';
return;
}
com_data.common_config.format = com_data.is_telephone === '1' ? 'telephone-number' : 'phone-number';
return get_format_checks(com_data, com_data.form_value_code, true);
},
//
filtered_Data(children) {
const componentMap = new Map(children.map((item) => [item.id, item]));
//
const list = children.filter((item) => ['single-text', 'select', 'radio-btns'].includes(item.key) && ['select', 'radio-btns'].includes(item.com_data.type) && item.com_data.show_hidden_list.length > 0);
const list_map = list.map((item) => ({ id: item.id, list: item.com_data.show_hidden_list }));
return children.filter((item) => {
//
if (item.is_enable !== '1') return false;
if (list_map.length === 0) return true;
//
const isShownByRule = list_map.some((list_item) => {
const targetComponent = componentMap.get(list_item.id);
//
if (!targetComponent) return false;
return list_item.list.some(hidden_item => {
//
if (hidden_item.is_show.includes(item.id)) {
return targetComponent.com_data.form_value.includes(hidden_item.value);
} else {
return true;
}
});
});
return isShownByRule;
});
},
z_index_change(e) {
this.setData({
z_index_id: e
});
this.$emit('zIndexChange', e);
}
}
}
</script>
<style lang="scss" scoped>
.row-item {
padding: 10rpx 15rpx;
border-bottom: 2rpx solid #eee;
overflow: hidden;
}
.row-item:last-child {
border-bottom: none;
}
.column-item {
padding: 20rpx 15rpx;
padding-bottom: 20rpx;
}
.item_error {
background: #fef6e6;
}
.field-invalid-info {
color: #FF5353;
font-size: 24rpx;
line-height: 40rpx;
}
.required {
color: #FF5353;
font-weight: 700;
padding-left: 6rpx;
}
</style>

View File

@ -0,0 +1,248 @@
<template>
<view class="wh-auto flex-col gap-5 oh">
<!-- 输入框 -->
<view v-if="['single-text', 'radio-btns', 'select'].includes(data_item.key) && data_item.com_data.type == 'single-text'" :style="data_item.com_data.common_style">
<component-input :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-input>
</view>
<!-- 多行输入框 -->
<view v-else-if="data_item.key == 'multi-text'" :style="data_item.com_data.common_style + 'padding: 18rpx 22rpx;'">
<component-textarea :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-textarea>
</view>
<!-- 复选按钮组 -->
<view v-else-if="['select-multi', 'checkbox'].includes(data_item.key) && data_item.com_data.type == 'checkbox' && propDirection !== 'row'">
<component-checkbox :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @dataOptionChange="data_option_change"></component-checkbox>
</view>
<!-- 单选按钮组 -->
<view v-else-if="['single-text', 'radio-btns', 'select'].includes(data_item.key) && data_item.com_data.type == 'radio-btns' && propDirection !== 'row'">
<component-radio :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-radio>
</view>
<!-- 下拉框 -->
<view v-else-if="(['single-text', 'radio-btns', 'select'].includes(data_item.key) && data_item.com_data.type == 'select') || (['single-text', 'radio-btns', 'select'].includes(data_item.key) && data_item.com_data.type == 'radio-btns' && propDirection == 'row')" :style="data_item.com_data.common_style">
<component-select :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propDirection="propDirection" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-select>
</view>
<!-- 下拉复选框 -->
<view v-else-if="(['select-multi', 'checkbox'].includes(data_item.key) && data_item.com_data.type == 'select-multi') || (['select-multi', 'checkbox'].includes(data_item.key) && data_item.com_data.type == 'checkbox' && propDirection == 'row')" :style="data_item.com_data.common_style">
<component-select-multi :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propDirection="propDirection" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @dataOptionChange="data_option_change" @zIndexChange="z_index_change"></component-select-multi>
</view>
<!-- 数字 -->
<view v-else-if="data_item.key == 'number'" :style="data_item.com_data.common_style">
<component-number :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-number>
</view>
<!-- 时间选择器 -->
<view v-else-if="data_item.key == 'date'" :style="data_item.com_data.common_style">
<component-date :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-date>
</view>
<!-- 时间选择器组 -->
<view v-else-if="data_item.key == 'date-group'" :style="data_item.com_data.common_style">
<component-date-group :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-date-group>
</view>
<!-- 地址选择器 -->
<view v-else-if="data_item.key == 'address'">
<component-address :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change" @regionEvent="region_event" @dataAddressChange="data_address_change" @zIndexChange="z_index_change"></component-address>
</view>
<!-- 手机 -->
<view v-else-if="data_item.key == 'phone'">
<component-phone :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-phone>
</view>
<!-- 密码 -->
<view v-else-if="data_item.key == 'pwd'" :style="data_item.com_data.common_style">
<component-pwd :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change"></component-pwd>
</view>
<!-- 评分 -->
<view v-else-if="data_item.key == 'score'">
<component-score :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change"></component-score>
</view>
<!-- 图片 -->
<view v-else-if="data_item.key == 'img'">
<component-image :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-image>
</view>
<!-- 视频 -->
<view v-else-if="data_item.key == 'video'">
<component-video :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-video>
</view>
<!-- 文本 -->
<view v-else-if="data_item.key == 'text'">
<component-text :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-text>
</view>
<!-- 文件 -->
<view v-else-if="data_item.key == 'attachments'">
<component-attachments :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-attachments>
</view>
<!-- 分割线 -->
<view v-else-if="data_item.key == 'auxiliary-line'">
<component-auxiliary-line :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-auxiliary-line>
</view>
<!-- 上传视频或图片 -->
<view v-else-if="['upload-img', 'upload-video'].includes(data_item.key)" :class="propDirection == 'row' ? 'padding-vertical-sm' : ''">
<component-upload :propValue="data_item.com_data" :propType="data_item.key == 'upload-img' ? 'img' : ( data_item.key == 'upload-video' ? 'video' : 'file')" :propKey="propKey" :data_itemFormId="data_itemFormId" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataChange="data_change"></component-upload>
</view>
<!-- 定位 -->
<view v-else-if="data_item.key == 'position'">
<component-position :propValue="data_item.com_data" :propKey="propKey" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataChange="data_change"></component-position>
</view>
<!-- #ifdef H5 || MP-WEIXIN || MP-QQ -->
<!-- 上传文件 -->
<view v-else-if="data_item.key == 'upload-attachments'">
<component-upload :propValue="data_item.com_data" propType="file" :propKey="propKey" :data_itemFormId="data_itemFormId" :propDataId="data_item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataChange="data_change"></component-upload>
</view>
<!-- #endif -->
</view>
</template>
<script>
import { isEmpty } from '@/common/js/common/common.js';
import componentInput from '@/pages/form-input/components/form-input/input.vue';
import componentTextarea from '@/pages/form-input/components/form-input/textarea.vue';
import componentCheckbox from '@/pages/form-input/components/form-input/checkbox.vue';
import componentRadio from '@/pages/form-input/components/form-input/radio.vue';
import componentSelect from '@/pages/form-input/components/form-input/select.vue';
import componentNumber from '@/pages/form-input/components/form-input/number.vue';
import componentDate from '@/pages/form-input/components/form-input/date.vue';
import componentDateGroup from '@/pages/form-input/components/form-input/date-group.vue';
import componentAddress from '@/pages/form-input/components/form-input/address.vue';
import componentSelectMulti from '@/pages/form-input/components/form-input/select-multi.vue';
import componentPhone from '@/pages/form-input/components/form-input/phone.vue';
import componentPwd from '@/pages/form-input/components/form-input/pwd.vue';
import componentScore from '@/pages/form-input/components/form-input/score.vue';
import componentImage from '@/pages/form-input/components/form-input/image.vue';
import componentVideo from '@/pages/form-input/components/form-input/video.vue';
import componentText from '@/pages/form-input/components/form-input/text.vue';
import componentAttachments from '@/pages/form-input/components/form-input/attachments.vue';
import componentAuxiliaryLine from '@/pages/form-input/components/form-input/auxiliary-line.vue';
import componentRichText from '@/pages/form-input/components/form-input/rich-text.vue';
import componentUpload from '@/pages/form-input/components/form-input/upload.vue';
import componentPosition from '@/pages/form-input/components/form-input/position.vue';
import componentRectOrRound from '@/pages/form-input/components/form-input/rect-or-round.vue';
import componentSubform from '@/pages/form-input/components/form-input/subform.vue';
export default {
name: 'formInput',
components: {
componentInput,
componentTextarea,
componentCheckbox,
componentRadio,
componentNumber,
componentSelect,
componentSelectMulti,
componentDate,
componentDateGroup,
componentAddress,
componentPhone,
componentPwd,
componentScore,
componentImage,
componentVideo,
componentText,
componentAttachments,
componentAuxiliaryLine,
componentRichText,
componentUpload,
componentPosition,
componentRectOrRound,
componentSubform
},
props: {
propData: {
type: Object,
default: () => {},
},
propDataFormId: {
type: [String, Number],
default: '',
},
propKey: {
type: [String, Number],
default: 0,
},
propDirection: {
type: String,
default: 'row',
},
propMobile: {
type: Object,
default: () => {},
},
propComponentStyle: {
type: String,
default: '',
},
propIndex: {
type: Number,
default: 0,
},
},
data() {
return {
data_item: {}
}
},
watch: {
propData: {
handler(newVal) {
this.init();
},
deep: true
},
propKey(val) {
this.init();
}
},
mounted() {
this.init();
},
methods: {
isEmpty,
init() {
this.setData({
data_item: this.propData,
})
},
data_change(e) {
this.$emit('dataChange', e, this.propIndex);
},
data_check(e) {
this.$emit('dataCheck', e, this.propIndex);
},
data_option_change(e) {
this.$emit('dataOptionChange', e, this.propIndex);
},
open_region(e) {
this.$emit('openRegion', e, this.propIndex);
},
region_event(e) {
this.$emit('regionEvent', e, this.propIndex);
},
z_index_change(e) {
this.$emit('zIndexChange', e, this.propIndex);
}
}
}
</script>
<style lang="scss" scoped>
.row-item {
padding: 10rpx 15rpx;
border-bottom: 2rpx solid #eee;
overflow: hidden;
}
.row-item:last-child {
border-bottom: none;
}
.column-item {
padding: 20rpx 15rpx;
padding-bottom: 20rpx;
}
.item_error {
background: #fef6e6;
}
.field-invalid-info {
color: #FF5353;
font-size: 24rpx;
line-height: 40rpx;
}
.required {
color: #FF5353;
font-weight: 700;
padding-left: 6rpx;
}
</style>

View File

@ -0,0 +1,252 @@
<template>
<view>
<uni-popup ref="popup" type="bottom" background-color="#fff" :animation="true" @maskClick="maskClick">
<view class="popup-content">
<view >
<view class="headBox padding-main">
<view @tap="close"><text class="text-size-sm">取消</text></view>
<!-- <view class="uni-page-head-title" v-if="titleShow">{{timeTitle}}</view> -->
<view><text class="text-size-sm cr-blue" @tap="submit_event"></text></view>
</view>
<picker-view :indicator-style="indicatorStyle" :value="value" @change="bindChange">
<picker-view-column v-for="(arr, n) in dateTimeArr" :key="n">
<view class="item flex-row align-c jc-c" v-for="(obj, index) in arr" :key="index">{{obj}}{{ dataType == 'date' ? dateUnitArr[n] || '' : timeUnitArr[n] || '' }}</view>
</picker-view-column>
</picker-view>
</view>
</view>
</uni-popup>
</view>
</template>
<script>
export default {
name: 'MyDatetime',
props: {
titleShow: {
type: Boolean,
default: true
},
timestr: {
//
type: String,
default: ''
},
dataType: {
type: String,
default: 'date'
},
//
shownum: {
type: [String, Number],
default: 3
}
},
data() {
const date = new Date()
const years = []
const year = date.getFullYear()
const months = []
const month = date.getMonth() + 1
const days = []
const day = date.getDate()
for (let i = 1970, len = date.getFullYear() + 100; i <= len; i++) {
years.push(i)
}
for (let i = 1; i <= 12; i++) {
months.push(i < 10 ? ('0'+ i.toString()) : i)
}
for (let i = 1; i <= 31; i++) {
days.push(i < 10 ? ('0'+ i.toString()) : i)
}
const hourArr = []
for (let i = 0; i < 24; i++) {
hourArr.push(i < 10 ? ('0'+ i.toString()) : i)
}
//
const minSecond = []
for (let i = 0; i <= 59; i++) {
minSecond.push(i < 10 ? ('0'+ i.toString()) : i)
}
return {
thirtyOne: days,
dateArr: [years, months, days ],
timeArr: [hourArr, minSecond, minSecond ],
dateUnitArr: [ '年', '月', '日', '时', '分', '秒'],
timeUnitArr: [ '时', '分', '秒'],
obj: { 0: 'year', 1: 'month', 2: 'day', 3: 'hour', 4: 'minute', 5: 'second' },
years,
months,
days,
hourArr,
minArr: minSecond,
secondArr: minSecond,
year,
month,
day,
hour: '00',
minute: '00',
second: '00',
value: [ year - 1970, month - 1, day - 1],
visible: true,
indicatorStyle: `height: 100rpx;`,
timer: null
}
},
computed: {
//
dateTimeArr () {
var num = parseInt(this.shownum) || 3
if (num <= 3 && num > 0) {
if (this.dataType == 'date') {
return this.dateArr.slice(0, num)
} else {
return this.timeArr.slice(0, num)
}
}
return this.dateArr
},
},
watch: {
//
month () {
this.countDay()
},
year () {
this.countDay()
}
},
created() {
this.countDay();
},
destroyed() {
this.timer && clearTimeout(this.timer)
},
methods: {
//
countDay () {
var index = new Date(this.year, parseInt(this.month), 0).getDate()
var days = this.thirtyOne.slice(0, index)
this.dateArr.splice(2, 1, days)
this.days = days
},
// timeStr: 2020-01-01 00:00:00
timeFill (timeStr) {
//
let dateObj = new Date();
//
if (timeStr) {
const time = this.dataType == 'date' ? timeStr : '1970-01-01 ' + timeStr.replace(/时|分|秒/g, ':').replace(/:+$/, ''); // ;
dateObj = new Date(time.replace(/-/g,'/').replace(/年|月|日/g, '/').replace(/\/+$/, ''));
}
const timeObj = {
year: dateObj.getFullYear(),
month: dateObj.getMonth() + 1,
day: dateObj.getDate(),
h: dateObj.getHours(),
m: dateObj.getMinutes(),
s: dateObj.getSeconds()
}
this.timer = setTimeout(() => {
if (this.dataType == 'date') {
this.value = [ (timeObj.year - 1970) || 0, timeObj.month - 1, timeObj.day - 1]
this.year = this.years[this.value[0]]
this.month = this.months[this.value[1]]
this.day = this.days[this.value[2]]
} else {
this.value = [ timeObj.h, timeObj.m, timeObj.s ]
this.hour = this.hourArr[this.value[0]]
this.minute = this.minArr[this.value[1]]
this.second = this.secondArr[this.value[2]]
}
}, 100)
},
//
submit_event() {
var num = parseInt(this.shownum) || 3
if (num <= 0 || num > 3) {
num = 3
}
let date = '';
if (this.dataType == 'date') {
if (num == 1) {
date = this.year;
} else if (num == 2) {
date = this.year + '/' + this.month;
} else if (num == 3) {
date = this.year + '/' + this.month + '/' + this.day;
}
} else {
if (num == 1) {
date = this.hour;
} else if (num == 2) {
date = this.hour + ':' + this.minute;
} else if (num == 3) {
date = this.hour + ':' + this.minute + ':' + this.second;
}
}
this.$emit('timeSubmit', date)
this.close()
},
//
open (timeStr) {
this.$nextTick(res => {
this.timeFill(timeStr || this.timestr || '')
this.$refs.popup.open()
})
},
/**
* 关闭窗口
*/
close () {
// close
this.$refs.popup.close()
},
bindChange(e) {
const val = e.detail.value
if (this.dataType == 'date') {
this.year = this.years[val[0]]
this.month = this.months[val[1]]
this.day = this.days[val[2]];
} else {
this.hour = this.hourArr[val[0] || 0]
this.minute = this.minArr[val[1] || 0]
this.second = this.secondArr[val[2] || 0]
}
},
maskClick() {
this.$emit('maskClick')
}
}
}
</script>
<style scoped>
.popup-content{
background-color: #FFFFFF;
}
.uni-page-head-title {
display: inline-block;
padding: 0 20rpx;
font-size: 26rpx;
height: 88rpx;
line-height: 88rpx;
color: #BEBEBE;
box-sizing: border-box;
}
.headBox{
display: flex;
justify-content: space-between;
}
picker-view {
width: 100%;
height: 600rpx;
margin-top:20rpx;
}
.item {
/* line-height: 100rpx; */
height: 100rpx;
text-align: center;
font-size: 30rpx;
}
</style>

View File

@ -0,0 +1,276 @@
<template>
<view class="wh-auto ht-auto pr">
<view v-for="(item, index) in data_list" :key="index" :class="(propDirection == 'row' ? 'row-item ' : 'column-item mb-10 ') + (item.com_data.common_config.is_error == '1' ? ' item_error' : '')" :data-id="item.id" :data-location-x="item.location.x" :data-location-y="item.location.y" :style="(item.key == 'auxiliary-line' ? 'border-bottom: 0rpx; ' : '')">
<view :class="'wh-auto ht-auto ' + (propDirection == 'row' ? (['video', 'img', 'upload-img', 'upload-video', 'multi-text'].includes(item.key) ? 'flex-row align-s gap-10' : 'flex-row align-b gap-10') : 'flex-col gap-10')">
<view v-if="!['auxiliary-line', 'upload-attachments', 'subform'].includes(item.key)" class="field-label flex-row align-c gap-10" :style="propFieldLabelStyle + (propDirection == 'row' && ['upload-img', 'upload-video'].includes(item.key) ? 'padding-top: 12rpx;line-height: 120rpx;' : '') + (propDirection == 'row' && ['multi-text'].includes(item.key) ? 'padding-top: 18rpx;' : '')">
<view class="flex-row align-c" :style="propTitleStyle">{{ item.com_data.title }}<view v-if="item.com_data.is_required == '1'" class="required">*</view></view>
<view v-if="item.com_data.common_config.help_is_show == '1' && !isEmpty(item.com_data.common_config.help_explain)" :data-value="item.com_data.common_config.help_explain" @tap="help_icon_event">
<iconfont name="icon-miaosha-hdgz" :size="propHelpIconStyle" color="#999"></iconfont>
</view>
</view>
<view class="flex-1 wh-auto ht-auto flex-col gap-5 oh">
<!-- 输入框 -->
<view v-if="['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'single-text'" :style="item.com_data.common_style">
<component-input :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-input>
</view>
<!-- 多行输入框 -->
<view v-else-if="item.key == 'multi-text'" :style="item.com_data.common_style + 'padding: 18rpx 22rpx;'">
<component-textarea :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-textarea>
</view>
<!-- 复选按钮组 -->
<view v-else-if="['select-multi', 'checkbox'].includes(item.key) && item.com_data.type == 'checkbox' && propDirection !== 'row'">
<component-checkbox :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @dataOptionChange="data_option_change"></component-checkbox>
</view>
<!-- 单选按钮组 -->
<view v-else-if="['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'radio-btns' && propDirection !== 'row'">
<component-radio :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-radio>
</view>
<!-- 下拉框 -->
<view v-else-if="(['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'select') || (['single-text', 'radio-btns', 'select'].includes(item.key) && item.com_data.type == 'radio-btns' && propDirection == 'row')" :style="item.com_data.common_style">
<component-select :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propDirection="propDirection" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-select>
</view>
<!-- 下拉复选框 -->
<view v-else-if="(['select-multi', 'checkbox'].includes(item.key) && item.com_data.type == 'select-multi') || (['select-multi', 'checkbox'].includes(item.key) && item.com_data.type == 'checkbox' && propDirection == 'row')" :style="item.com_data.common_style">
<component-select-multi :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propDirection="propDirection" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @dataOptionChange="data_option_change" @zIndexChange="z_index_change"></component-select-multi>
</view>
<!-- 数字 -->
<view v-else-if="item.key == 'number'" :style="item.com_data.common_style">
<component-number :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change"></component-number>
</view>
<!-- 时间选择器 -->
<view v-else-if="item.key == 'date'" :style="item.com_data.common_style">
<component-date :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-date>
</view>
<!-- 时间选择器组 -->
<view v-else-if="item.key == 'date-group'" :style="item.com_data.common_style">
<component-date-group :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-date-group>
</view>
<!-- 地址选择器 -->
<view v-else-if="item.key == 'address'">
<component-address :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change" @regionEvent="region_event" @dataAddressChange="data_address_change" @zIndexChange="z_index_change"></component-address>
</view>
<!-- 手机 -->
<view v-else-if="item.key == 'phone'">
<component-phone :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change" @zIndexChange="z_index_change"></component-phone>
</view>
<!-- 密码 -->
<view v-else-if="item.key == 'pwd'" :style="item.com_data.common_style">
<component-pwd :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change"></component-pwd>
</view>
<!-- 评分 -->
<view v-else-if="item.key == 'score'">
<component-score :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataCheck="data_check" @dataChange="data_change"></component-score>
</view>
<!-- 图片 -->
<view v-else-if="item.key == 'img'">
<component-image :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-image>
</view>
<!-- 视频 -->
<view v-else-if="item.key == 'video'">
<component-video :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-video>
</view>
<!-- 文本 -->
<view v-else-if="item.key == 'text'">
<component-text :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-text>
</view>
<!-- 文件 -->
<view v-else-if="item.key == 'attachments'">
<component-attachments :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-attachments>
</view>
<!-- 分割线 -->
<view v-else-if="item.key == 'auxiliary-line'">
<component-auxiliary-line :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection"></component-auxiliary-line>
</view>
<!-- 上传视频或图片 -->
<view v-else-if="['upload-img', 'upload-video'].includes(item.key)" :class="propDirection == 'row' ? 'padding-vertical-sm' : ''">
<component-upload :propValue="item.com_data" :propType="item.key == 'upload-img' ? 'img' : ( item.key == 'upload-video' ? 'video' : 'file')" :propKey="propKey" :propDataFormId="propDataFormId" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataChange="data_change"></component-upload>
</view>
<!-- 定位 -->
<view v-else-if="item.key == 'position'">
<component-position :propValue="item.com_data" :propKey="propKey" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataChange="data_change"></component-position>
</view>
<!-- #ifdef H5 || MP-WEIXIN || MP-QQ -->
<!-- 上传文件 -->
<view v-else-if="item.key == 'upload-attachments'">
<component-upload :propValue="item.com_data" propType="file" :propKey="propKey" :propDataFormId="propDataFormId" :propDataId="item.id" :propMobile="propMobile" :propStyle="propComponentStyle" :propDirection="propDirection" @dataChange="data_change"></component-upload>
</view>
<!-- #endif -->
<view v-if="!isEmpty(item.com_data.common_config.error_text)" class="field-invalid-info">{{ item.com_data.common_config.error_text }}</view>
</view>
</view>
</view>
</view>
</template>
<script>
import { isEmpty } from '@/common/js/common/common.js';
import componentInput from '@/pages/form-input/components/form-input/input.vue';
import componentTextarea from '@/pages/form-input/components/form-input/textarea.vue';
import componentCheckbox from '@/pages/form-input/components/form-input/checkbox.vue';
import componentRadio from '@/pages/form-input/components/form-input/radio.vue';
import componentSelect from '@/pages/form-input/components/form-input/select.vue';
import componentNumber from '@/pages/form-input/components/form-input/number.vue';
import componentDate from '@/pages/form-input/components/form-input/date.vue';
import componentDateGroup from '@/pages/form-input/components/form-input/date-group.vue';
import componentAddress from '@/pages/form-input/components/form-input/address.vue';
import componentSelectMulti from '@/pages/form-input/components/form-input/select-multi.vue';
import componentPhone from '@/pages/form-input/components/form-input/phone.vue';
import componentPwd from '@/pages/form-input/components/form-input/pwd.vue';
import componentScore from '@/pages/form-input/components/form-input/score.vue';
import componentImage from '@/pages/form-input/components/form-input/image.vue';
import componentVideo from '@/pages/form-input/components/form-input/video.vue';
import componentText from '@/pages/form-input/components/form-input/text.vue';
import componentAttachments from '@/pages/form-input/components/form-input/attachments.vue';
import componentAuxiliaryLine from '@/pages/form-input/components/form-input/auxiliary-line.vue';
import componentRichText from '@/pages/form-input/components/form-input/rich-text.vue';
import componentUpload from '@/pages/form-input/components/form-input/upload.vue';
import componentPosition from '@/pages/form-input/components/form-input/position.vue';
import componentRectOrRound from '@/pages/form-input/components/form-input/rect-or-round.vue';
import componentSubform from '@/pages/form-input/components/form-input/subform.vue';
export default {
name: 'formInput',
components: {
componentInput,
componentTextarea,
componentCheckbox,
componentRadio,
componentNumber,
componentSelect,
componentSelectMulti,
componentDate,
componentDateGroup,
componentAddress,
componentPhone,
componentPwd,
componentScore,
componentImage,
componentVideo,
componentText,
componentAttachments,
componentAuxiliaryLine,
componentRichText,
componentUpload,
componentPosition,
componentRectOrRound,
componentSubform
},
props: {
propValue: {
type: Array,
default: () => [],
},
propFieldLabelStyle: {
type: String,
default: '',
},
propTitleStyle: {
type: String,
default: '',
},
propHelpIconStyle: {
type: String,
default: '',
},
propDataFormId: {
type: [String, Number],
default: '',
},
propKey: {
type: [String, Number],
default: 0,
},
propDirection: {
type: String,
default: 'row',
},
propMobile: {
type: Object,
default: () => {},
},
propComponentStyle: {
type: String,
default: '',
},
propIndex: {
type: Number,
default: 0,
}
},
data() {
return {
data_list: [],
}
},
watch: {
propValue: {
handler(newVal) {
this.init();
},
deep: true
},
propKey(val) {
this.init();
}
},
mounted() {
this.init();
},
methods: {
isEmpty,
init() {
this.setData({
data_list: this.propValue,
})
},
help_icon_event(e) {
this.$emit('helpIconEvent', e.currentTarget.dataset.value);
},
data_change(e) {
this.$emit('dataChange', e, this.propIndex);
},
data_check(e) {
this.$emit('dataCheck', e, this.propIndex);
},
data_option_change(e) {
this.$emit('dataOptionChange', e, this.propIndex);
},
open_region(e) {
this.$emit('openRegion', e, this.propIndex);
},
region_event(e) {
this.$emit('regionEvent', e, this.propIndex);
},
z_index_change(e) {
this.$emit('zIndexChange', e);
}
}
}
</script>
<style lang="scss" scoped>
.row-item {
padding: 10rpx 15rpx;
border-bottom: 2rpx solid #eee;
overflow: hidden;
}
.row-item:last-child {
border-bottom: none;
}
.column-item {
padding: 20rpx 15rpx;
padding-bottom: 20rpx;
}
.item_error {
background: #fef6e6;
}
.field-invalid-info {
color: #FF5353;
font-size: 24rpx;
line-height: 40rpx;
}
.required {
color: #FF5353;
font-weight: 700;
padding-left: 6rpx;
}
</style>