schangxiang@126.com
2025-09-19 fc752b66a7976188c4edd5e3fb7ca6bb2822e441
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
<template>
  <el-form-item :label="widget.label" :prop="widget.model">
    <template v-if="widget.type == 'input'">
      <el-input v-if="widget.options.dataType == 'number' || widget.options.dataType == 'integer' || widget.options.dataType == 'float'" v-model.number="dataModel" :type="widget.options.dataType" :placeholder="widget.options.placeholder" :style="{width: widget.options.width}"></el-input>
      <el-input v-else v-model="dataModel" :type="widget.options.dataType" :placeholder="widget.options.placeholder" :style="{width: widget.options.width}"></el-input>
    </template>
 
    <template v-if="widget.type == 'textarea'">
      <el-input v-model="dataModel" :rows="5" :placeholder="widget.options.placeholder" :style="{width: widget.options.width}" type="textarea"></el-input>
    </template>
 
    <template v-if="widget.type == 'number'">
      <el-input-number v-model="widget.options.defaultValue" :style="{width: widget.options.width}" :step="widget.options.step" controls-position="right"></el-input-number>
    </template>
 
    <template v-if="widget.type == 'radio'">
      <el-radio-group v-model="dataModel" :style="{width: widget.options.width}">
        <el-radio v-for="(item, index) in (widget.options.remote ? widget.options.remoteOptions : widget.options.options)" :key="index" :style="{display: widget.options.inline ? 'inline-block' : 'block'}" :label="item.value">
          <template v-if="widget.options.remote">{{ item.label }}</template>
          <template v-else>{{ widget.options.showLabel ? item.label : item.value }}</template>
        </el-radio>
      </el-radio-group>
    </template>
 
    <template v-if="widget.type == 'checkbox'">
      <el-checkbox-group v-model="dataModel" :style="{width: widget.options.width}">
        <el-checkbox v-for="(item, index) in (widget.options.remote ? widget.options.remoteOptions : widget.options.options)" :key="index" :style="{display: widget.options.inline ? 'inline-block' : 'block'}" :label="item.value">
          <template v-if="widget.options.remote">{{ item.label }}</template>
          <template v-else>{{ widget.options.showLabel ? item.label : item.value }}</template>
        </el-checkbox>
      </el-checkbox-group>
    </template>
 
    <template v-if="widget.type == 'cascader'">
      <el-cascader v-model="dataModel" :placeholder="widget.options.placeholder" :options="widget.options.options" :style="{width: widget.options.width}" filterable change-on-select></el-cascader>
    </template>
 
    <template v-if="widget.type == 'time'">
      <el-time-picker v-model="dataModel" :is-range="widget.options.isRange" :placeholder="widget.options.placeholder" :start-placeholder="widget.options.startPlaceholder" :end-placeholder="widget.options.endPlaceholder" :readonly="widget.options.readonly" :disabled="widget.options.disabled" :editable="widget.options.editable" :clearable="widget.options.clearable" :arrow-control="widget.options.arrowControl" :value-format="widget.options.format" :style="{width: widget.options.width}">
      </el-time-picker>
    </template>
 
    <template v-if="widget.type=='date'">
      <el-date-picker v-model="dataModel" :type="widget.options.type" :placeholder="widget.options.placeholder" :start-placeholder="widget.options.startPlaceholder" :end-placeholder="widget.options.endPlaceholder" :readonly="widget.options.readonly" :disabled="widget.options.disabled" :editable="widget.options.editable" :clearable="widget.options.clearable" :value-format="widget.options.timestamp ? 'timestamp' : widget.options.format" :format="widget.options.format" :style="{width: widget.options.width}">
      </el-date-picker>
    </template>
 
    <template v-if="widget.type =='rate'">
      <el-rate v-model="dataModel" :max="widget.options.max" :disabled="widget.options.disabled" :allow-half="widget.options.allowHalf"></el-rate>
    </template>
 
    <template v-if="widget.type == 'color'">
      <el-color-picker v-model="dataModel" :disabled="widget.options.disabled" :show-alpha="widget.options.showAlpha"></el-color-picker>
    </template>
 
    <template v-if="widget.type == 'select'">
      <el-select v-model="dataModel" :disabled="widget.options.disabled" :multiple="widget.options.multiple" :clearable="widget.options.clearable" :placeholder="widget.options.placeholder" :style="{width: widget.options.width}">
        <el-option v-for="item in (widget.options.remote ? widget.options.remoteOptions : widget.options.options)" :key="item.value" :value="item.value" :label="widget.options.showLabel || widget.options.remote?item.label:item.value"></el-option>
      </el-select>
    </template>
 
    <template v-if="widget.type=='switch'">
      <el-switch v-model="dataModel" :disabled="widget.options.disabled">
      </el-switch>
    </template>
 
    <template v-if="widget.type=='slider'">
      <el-slider v-model="dataModel" :min="widget.options.min" :max="widget.options.max" :disabled="widget.options.disabled" :step="widget.options.step" :show-input="widget.options.showInput" :range="widget.options.range" :style="{width: widget.options.width}"></el-slider>
    </template>
 
    <template v-if="widget.type=='imgupload'">
      <fm-upload v-model="dataModel" :disabled="widget.options.disabled" :style="{'width': widget.options.width}" :width="widget.options.size.width" :height="widget.options.size.height" :token="widget.options.token" :domain="widget.options.domain">
      </fm-upload>
    </template>
  </el-form-item>
</template>
 
<script>
import FmUpload from "./Upload";
 
export default {
  components: {
    FmUpload
  },
  props: {
    widget: {
      type: Object,
      default: () => {
        return {};
      }
    },
    models: {
      type: Object,
      default: () => {
        return {};
      }
    },
    rules: {
      type: Object,
      default: () => {
        return {};
      }
    },
    remote: {
      type: Object,
      default: () => {
        return {};
      }
    }
  },
  data() {
    return {
      dataModel: this.models[this.widget.model]
    };
  },
  watch: {
    dataModel: {
      deep: true,
      handler(val) {
        this.models[this.widget.model] = val;
        this.$emit("update:models", {
          ...this.models,
          [this.widget.model]: val
        });
      }
    },
    models: {
      deep: true,
      handler(val) {
        // console.log('--------')
        this.dataModel = val[this.widget.model];
      }
    }
  },
  created() {
    if (
      this.widget.options.remote &&
      this.remote[this.widget.options.remoteFunc]
    ) {
      this.remote[this.widget.options.remoteFunc](data => {
        this.widget.options.remoteOptions = data.map(item => {
          return {
            value: item[this.widget.options.props.value],
            label: item[this.widget.options.props.label]
          };
        });
      });
    }
 
    if (this.widget.type === "imgupload") {
      this.remote[this.widget.options.tokenFunc](data => {
        this.widget.options.token = data;
      });
    }
  }
};
</script>