@ -0,0 +1,4 @@ |
|||
export default { |
|||
flowSave: 'flow/save', |
|||
flowList: 'flow/list' |
|||
} |
|||
@ -0,0 +1,19 @@ |
|||
export default { |
|||
designSave: 'design/save', // 保存设计的表单
|
|||
designEdit: 'design/edit', // 保存设计的表单
|
|||
designList: 'design/list', // 获取所有已设计的表单
|
|||
designChange: 'design/change', // 改变设计表单部分设置内容
|
|||
designDelete: 'design/delete', // 根据id删除已设计
|
|||
designById: 'design/id', // 根据id获取已设计的表单
|
|||
sourceList: 'dataSource/list', // 获取数据源列表
|
|||
sourceDelete: 'dataSource/delete', |
|||
sourceCreat: 'dataSource/creat', // 创建数据源
|
|||
sourceEdit: 'dataSource/edit', // 编辑数据源
|
|||
sourceById: 'dataSource/id', // 根据id获取数据源
|
|||
saveFormContent: 'content/save', // 往已设计好的表单里添加内容
|
|||
editFormContent: 'content/edit', // 往已设计好的表单里添加内容
|
|||
getContentList: 'content/list', // 获取表单内容列表
|
|||
delFormContent: 'content/delete', // 删除表单内容
|
|||
getFormContent: 'content/id', // 获取表单内容
|
|||
upload: 'upload/single' // 默认el-upload上传url,也可单独在设计时填写上传地址
|
|||
} |
|||
@ -0,0 +1,73 @@ |
|||
import request from '@/utils/DesignForm/request' |
|||
import form from './form' |
|||
import system from './system' |
|||
import flow from './flow' |
|||
const allApi: any = Object.assign(form, system, flow) |
|||
export const getRequest = (apiKey: string, data?: any, options: any = {}) => { |
|||
let url = allApi[apiKey] || apiKey |
|||
// 解决动态url 如/api/delete/id(id为动态时)
|
|||
// url设置为:/api/delete/$id
|
|||
// options参数设置:options:{apikey:$id:xx}
|
|||
if (Object.keys(options.apiKey || {}).length) { |
|||
for (const key in options.apiKey) { |
|||
url = url.replace(key, options.apiKey[key]) |
|||
} |
|||
} |
|||
let obj: any = Object.assign( |
|||
{ |
|||
url: '/api/' + url, // 添加个前缀
|
|||
method: 'POST', |
|||
data |
|||
}, |
|||
options |
|||
) |
|||
// github演示时使用下面地址
|
|||
if (window.location.host.indexOf('github') !== -1) { |
|||
let id = '' |
|||
if (url.indexOf('/id') !== -1 && data.id) { |
|||
id = data.id + '' |
|||
} |
|||
if (url.indexOf('/id') !== -1 && data.formId) { |
|||
id += data.formId |
|||
} |
|||
if (url.indexOf('design/list') !== -1 && data.type) { |
|||
id = data.type |
|||
} |
|||
if (url.indexOf('content/list') !== -1 && data.formId) { |
|||
id = data.formId |
|||
} |
|||
if ( |
|||
url.includes('/save') || |
|||
url.includes('/edit') || |
|||
url.includes('/delete') || |
|||
url.includes('/creat') || |
|||
url.includes('/change') || |
|||
url.includes('/single') |
|||
) { |
|||
url = 'ok' |
|||
} |
|||
if (options.method) { |
|||
delete options.method |
|||
} |
|||
obj = Object.assign( |
|||
{ |
|||
url: `./mock/${url}${id}.json`, |
|||
method: 'GET', |
|||
params: data |
|||
}, |
|||
options |
|||
) |
|||
} |
|||
return request(obj) |
|||
} |
|||
export const uploadUrl = '/api/' + allApi.upload |
|||
/*export function uploadFiledTinymce(data, url) { |
|||
return request({ |
|||
url: url || '/upload/single', |
|||
method: 'post', |
|||
data, |
|||
headers: { |
|||
'Content-Type': 'multipart/form-data' |
|||
} |
|||
}) |
|||
}*/ |
|||
@ -0,0 +1,22 @@ |
|||
export default { |
|||
dictSave: 'system/dict/save', |
|||
dictList: 'system/dict/list', |
|||
dictDelete: 'system/dict/delete', |
|||
dictEdit: 'system/dict/edit', |
|||
deptSave: 'system/dept/save', |
|||
deptEdit: 'system/dept/edit', |
|||
deptList: 'system/dept/list', |
|||
deptDelete: 'system/dept/delete', |
|||
menuSave: 'system/menu/save', |
|||
menuEdit: 'system/menu/edit', |
|||
menuList: 'system/menu/list', |
|||
menuDelete: 'system/menu/delete', |
|||
roleSave: 'system/role/save', |
|||
roleEdit: 'system/role/edit', |
|||
roleList: 'system/role/list', |
|||
roleDelete: 'system/role/delete', |
|||
userSave: 'system/user/save', |
|||
userEdit: 'system/user/edit', |
|||
userList: 'system/user/list', |
|||
userDelete: 'system/user/delete' |
|||
} |
|||
@ -0,0 +1,87 @@ |
|||
export interface FormList { |
|||
name: string |
|||
type: string |
|||
control: any // 当前type表单控件所有props
|
|||
item?: FormItem // formItem组件所有props
|
|||
config: any // 其他配置信息
|
|||
customRules?: any // 自定义检验规则,这个规则会合并进item.rules
|
|||
columns?: any // 布局字段
|
|||
tableData?: any // 子表时
|
|||
options?: Options[] // radio,checkbox,select选项
|
|||
} |
|||
export interface Options { |
|||
label: string |
|||
value: string |
|||
disabled?: boolean |
|||
} |
|||
export interface FormItem { |
|||
label?: string |
|||
labelWidth?: string |
|||
required?: boolean |
|||
rules?: any |
|||
error?: string |
|||
showMessage?: boolean |
|||
inlineMessage?: boolean |
|||
size?: string |
|||
showLabel?: boolean |
|||
} |
|||
|
|||
export interface FormData { |
|||
list: FormList[] |
|||
form: any // form所有props
|
|||
config?: { |
|||
style?: string // 表单css样式,相当于scope
|
|||
hideField?: string[] // 使用v-if隐藏的字段,用于交互
|
|||
requestUrl?: string // 表单数据请求url
|
|||
addUrl?: string // 提交表单
|
|||
editUrl?: string // 编辑保存
|
|||
addLoad?: boolean |
|||
expand?: boolean // 用于设置筛选条件默认展开/收起状态
|
|||
} |
|||
events?: { |
|||
beforeRequest?: Function |
|||
afterResponse?: Function | string |
|||
beforeSubmit?: Function |
|||
afterSubmit?: Function |
|||
change?: Function |
|||
} |
|||
} |
|||
|
|||
export interface TableData { |
|||
tableProps?: any |
|||
columns: any |
|||
controlBtn?: any |
|||
events?: any |
|||
treeData?: { |
|||
show: boolean |
|||
beforeRequest?: Function |
|||
afterResponse?: Function | string |
|||
method: string |
|||
requestUrl: string |
|||
name: string |
|||
} |
|||
operateBtn?: any |
|||
config?: any |
|||
} |
|||
|
|||
export interface OpenDrawer { |
|||
//visible: boolean
|
|||
title?: string |
|||
direction?: string |
|||
content?: any |
|||
codeType?: string |
|||
callback?: Function | string |
|||
type?: string |
|||
} |
|||
|
|||
|
|||
//文件
|
|||
export interface FileAttribute{ |
|||
imgPath:any; |
|||
title:any; |
|||
formData:any; |
|||
} |
|||
export interface FileAttributeAll{ |
|||
visible:boolean; |
|||
list:FileAttribute[]; |
|||
} |
|||
@ -0,0 +1,108 @@ |
|||
import { EDITTYPE } from '@/utils/DesignForm/form' |
|||
|
|||
export const aceEdit = (data: any, id?: string, type?: string | undefined) => { |
|||
id = id || 'editJson' |
|||
type = type || EDITTYPE |
|||
// @ts-ignore
|
|||
const editor = ace.edit(id) |
|||
editor.setOptions({ |
|||
enableBasicAutocompletion: true, |
|||
enableSnippets: true, |
|||
enableLiveAutocompletion: true |
|||
}) |
|||
editor.setFontSize(14) |
|||
editor.setShowPrintMargin(false) |
|||
editor.session.setMode('ace/mode/' + type) |
|||
editor.setTheme('ace/theme/tomorrow_night') |
|||
editor.setValue(data) |
|||
return editor |
|||
} |
|||
// 将字符类数字转为数值类
|
|||
export const formatNumber = (val: any) => { |
|||
// 将字符类数字转为数值类
|
|||
if (typeof val === 'string' && /^\d+(\.\d+)?$/.test(val.toString())) { |
|||
// 为数字
|
|||
return Number(val) |
|||
} else { |
|||
return val |
|||
} |
|||
} |
|||
// 转所有值转为字符串类型
|
|||
export const formatToString = (val: any) => { |
|||
if (val !== undefined) { |
|||
return val.toString() |
|||
} else { |
|||
return val |
|||
} |
|||
} |
|||
// 将{key:value}转[{label:'key',value:'value'}]
|
|||
export const objectToArray = (obj: any) => { |
|||
if (Object.prototype.toString.call(obj) === '[object Object]') { |
|||
const temp: any = [] |
|||
for (const key in obj) { |
|||
temp.push({ |
|||
label: obj[key], |
|||
value: key |
|||
}) |
|||
} |
|||
return temp |
|||
} |
|||
return obj |
|||
} |
|||
/**** |
|||
* 动态插入移除css |
|||
* @param id 标签id |
|||
* @param cssContent 要插入的css内容 |
|||
* @param append true插入false移除 |
|||
*/ |
|||
export const appendOrRemoveStyle = ( |
|||
id: string, |
|||
cssContent: string, |
|||
append?: boolean |
|||
) => { |
|||
const styleId: any = document.getElementById(id) |
|||
if (styleId && append) { |
|||
// 存在时直接修改,不用多次插入
|
|||
styleId.innerText = cssContent |
|||
return |
|||
} |
|||
if (cssContent && append) { |
|||
const styleEl = document.createElement('style') |
|||
styleEl.id = id |
|||
styleEl.type = 'text/css' |
|||
styleEl.appendChild(document.createTextNode(cssContent)) |
|||
document.head.appendChild(styleEl) |
|||
} |
|||
if (!append || !cssContent) { |
|||
// 移除
|
|||
styleId && styleId.parentNode.removeChild(styleId) |
|||
} |
|||
} |
|||
// 定义两个空方法,用于在编辑事件时作为默认值
|
|||
export const beforeRequest = |
|||
'opt=(data, route) => {\n' + |
|||
' // data经过处理后返回\n' + |
|||
" console.log('beforeRequest',data)\n" + |
|||
' return data\n' + |
|||
'}' |
|||
export const afterResponse = |
|||
'opt=(res) => {\n' + |
|||
' // res返回数据\n' + |
|||
" console.log('afterResponse',res)\n" + |
|||
' return res\n' + |
|||
'}' |
|||
|
|||
export const onChange = |
|||
'opt=(key,model) => {\n' + |
|||
' // name当前改变组件的值,model表单的值\n' + |
|||
" console.log('onChange',key)\n" + |
|||
' return model\n' + |
|||
'}' |
|||
|
|||
// provide 方法定义的key
|
|||
const prefix = 'AK' |
|||
export const constControlChange = prefix + 'ControlChange' // 表单组件改变事件
|
|||
export const constSetFormOptions = prefix + 'SetFormOptions' // 使用setOptions设置下拉值
|
|||
export const constGetControlByName = prefix + 'GetControlByName' // 根据name从formData.list查找数据
|
|||
export const constFormBtnEvent = prefix + 'FormBtnEvent' // 按钮组件事件
|
|||
export const constFormProps = prefix + 'FormProps' // 按钮组件事件
|
|||
@ -0,0 +1,139 @@ |
|||
{ |
|||
"code": 0, |
|||
"msg": "成功", |
|||
"data": { |
|||
"childDepartments": [ |
|||
{ |
|||
"id": "103", |
|||
"departmentKey": "GK0302", |
|||
"departmentName": "IT", |
|||
"parentId": "102", |
|||
"departmentNames": "IT" |
|||
}, |
|||
{ |
|||
"id": "272", |
|||
"departmentKey": "GK0301", |
|||
"departmentName": "企管", |
|||
"parentId": "102", |
|||
"departmentNames": "企管" |
|||
}, |
|||
{ |
|||
"id": "273", |
|||
"departmentKey": "GK0303", |
|||
"departmentName": "岗监", |
|||
"parentId": "102", |
|||
"departmentNames": "岗监" |
|||
} |
|||
], |
|||
"employees": [ |
|||
{ |
|||
"id": "10638305678856192", |
|||
"employeeName": "庞富镇", |
|||
"isLeave": "0", |
|||
"open": "false", |
|||
"icon": "", |
|||
"iconToBase64": "", |
|||
"wechat": "PangFuZhen", |
|||
"departmentid": 102, |
|||
"departmentname": "企管部", |
|||
"postid": 1949, |
|||
"postname": "开发专员", |
|||
"tema": 0, |
|||
"temaname": "" |
|||
}, |
|||
{ |
|||
"id": "20833135155482624", |
|||
"employeeName": "薛中国", |
|||
"isLeave": "0", |
|||
"open": "false", |
|||
"icon": "", |
|||
"iconToBase64": "", |
|||
"wechat": "XueZhongGuo", |
|||
"departmentid": 102, |
|||
"departmentname": "企管部", |
|||
"postid": 797, |
|||
"postname": "IT", |
|||
"tema": 0, |
|||
"temaname": "" |
|||
}, |
|||
{ |
|||
"id": "95196156539179008", |
|||
"employeeName": "胡齐帅", |
|||
"isLeave": "0", |
|||
"open": "false", |
|||
"icon": "", |
|||
"iconToBase64": "", |
|||
"wechat": "", |
|||
"departmentid": 102, |
|||
"departmentname": "企管部", |
|||
"postid": 798, |
|||
"postname": "岗监员", |
|||
"tema": 0, |
|||
"temaname": "" |
|||
}, |
|||
{ |
|||
"id": "95196160066588672", |
|||
"employeeName": "高宇", |
|||
"isLeave": "0", |
|||
"open": "false", |
|||
"icon": "", |
|||
"iconToBase64": "", |
|||
"wechat": "", |
|||
"departmentid": 102, |
|||
"departmentname": "企管部", |
|||
"postid": 798, |
|||
"postname": "岗监员", |
|||
"tema": 0, |
|||
"temaname": "" |
|||
}, |
|||
{ |
|||
"id": "112229905093103616", |
|||
"employeeName": "孙殿泉", |
|||
"isLeave": "0", |
|||
"open": "false", |
|||
"icon": "", |
|||
"iconToBase64": "", |
|||
"wechat": "", |
|||
"departmentid": 102, |
|||
"departmentname": "企管部", |
|||
"postid": 1120, |
|||
"postname": "新员工", |
|||
"tema": 0, |
|||
"temaname": "" |
|||
}, |
|||
{ |
|||
"id": "134825671351341056", |
|||
"employeeName": "李文轩", |
|||
"isLeave": "0", |
|||
"open": "false", |
|||
"icon": "", |
|||
"iconToBase64": "\r\nHBwgJC4nICIsIxwcKDcpLDAxNDQ0Hyc5PTgyPC4zNDL/2wBDAQkJCQwLDBgNDRgyIRwhMjIyMjIy\r\nMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjL/wAARCAB4AGQDASIA\r\nAhEBAxEB/8QAHwAAAQUBAQEBAQEAAAAAAAAAAAECAwQFBgcICQoL/8QAtRAAAgEDAwIEAwUFBAQA\r\nAAF9AQIDAAQRBRIhMUEGE1FhByJxFDKBkaEII0KxwRVS0fAkM2JyggkKFhcYGRolJicoKSo0NTY3\r\nODk6Q0RFRkdISUpTVFVWV1hZWmNkZWZnaGlqc3R1dnd4eXqDhIWGh4iJipKTlJWWl5iZmqKjpKWm\r\np6ipqrKztLW2t7i5usLDxMXGx8jJytLT1NXW19jZ2uHi4+Tl5ufo6erx8vP09fb3+Pn6/8QAHwEA\r\nAwEBAQEBAQEBAQAAAAAAAAECAwQFBgcICQoL/8QAtREAAgECBAQDBAcFBAQAAQJ3AAECAxEEBSEx\r\nBhJBUQdhcRMiMoEIFEKRobHBCSMzUvAVYnLRChYkNOEl8RcYGRomJygpKjU2Nzg5OkNERUZHSElK\r\nU1RVVldYWVpjZGVmZ2hpanN0dXZ3eHl6goOEhYaHiImKkpOUlZaXmJmaoqOkpaanqKmqsrO0tba3\r\nuLm6wsPExcbHyMnK0tPU1dbX2Nna4uPk5ebn6Onq8vP09fb3+Pn6/9oADAMBAAIRAxEAPwD2M0ma\r\nU1GTUgOzRmoy1YfiDxTY+HYPMu5Bk8hR1NIZ0G6kJrze6+MvhyG3keFbqaRfux7Au4/UnpXMJ8dr\r\ngzvnRYzCfugT8j6nHP5U7Ae37qN1eOW/xzt9xFzpTckY8uUfj1rsvDvxG0HxC8cMc/2e6ccQzHBJ\r\n9AehoA7HdSZpAcj2NFADgaTOKSikAuaKKKBDyaiY09jioHbFAyC/u0s7Ka4c/LGhY/hXy34n1q41\r\nnWLiaSeRo952AngD0GK+hvGdw8XhDVXj+8tuxr5hs7a71O58mAdPvN6CmmtwtfREJ2luV4Hc0rtH\r\ngAsBXUR+DHdPmuHye9MbwK2f+PhvxFT7WHc09jPscqx3kBMGnxyNEyFWYMOcqeQa6xfAmSD55+mK\r\nbe+CPLhLQSN5gHfpR7WAexn2PRvhj8Rp9QC6VqkhlkRAIpT1IHY+p6V68DmvkTRL250HXonkUJLG\r\n3O7IGPw/nX1fp1wLrT7ecEESRqwI6ciqZmXc0DmkBzTlpCFxRS0UWGMeqkzVac1TnoAxtctP7R0a\r\n9syM+dC6D6kcfrXivg6xS2sruecKmJCpJ7Y617lO20E4zgZrzRLW2e3uJPJAhmmaby8Zxk5xWVR6\r\nWN6EW5XKlvqukzSCP7Rt9ypArdh06JowwkWTPcVhSajaQmKM2MxDnAIRfl+vORWxZSo0aGE4SQZA\r\nxj9K55JdDupu+jHz21tbJullWMDqScVkPq2kmfyhdKT06Vq38kUMZkmwVHBzzWTHqmlXKtGbN1Ab\r\nYzNEMZoik0E3Z2Ry/jDR999aTW6585hGCvcnpX0Pp0H2PTra2znyYljznOcDFeXpa2azadLcKxt7\r\nW4WbaoyTtDEfrj8q9TgkEsSSLkK6hhketdVKXu2OCvFqVy2pqRTUCmpVNaGJLRQOlFAELmqk3SrL\r\nmq0vIpAZ83WuOuoY7a6kiUcBiQPTPP8AWuxnFcrrkey8Eg/iXn6j/IrGotDow7tKxnvFEVLMOBUV\r\nvMrTFlwQvAA7U2SXdGwJ7VlC+nhmVUEawom0rjlj65/piudK56N0jbeWKS4aKQDB7GpktYowcKMf\r\nSsKC4uJ5XEuzBYMhAwVH1zz+lbQuMRgHJNFrBo1cs2NiuoX8UBXMe7Lj/Zwa9AQAAAcCuT8LRbri\r\n4mP8KhR+PP8ASusTpXTSVkebiJXlbsSipFqNakWtjAlB4opo6UUAQsahenu1Qs1ICtOvBrmtdiL2\r\n/mDqh/Q/5FdLMw2msS/eNYpDIwCYOcniokrqxcHZpnC3Tt5bKhwx9aoP56jLyRAHtsOKumRWJD8H\r\n9DTHWNwd2OOxrlR6Ka3K4+1smVeIjt8ladtKXjUH7w61WVkVQAOPapYLiNbiM7fkDAscds0A5JbH\r\nomg2ptdPQMMPJ87fj0/SttKzreZXRXRgynkEHg1djfIrrjoebJtu7LIqRaiVqlU1ZJIOlFA6UUAZ\r\nzyY71VluAookJPU1CF5yKzuOww75m5+UVR1/TjcaBN5K5kjYSnHUgdf0OfwrTAxzU8b4ak9Slo7n\r\nlPlh/SgQuBhG6diMiuq1zwnL5jXWkgMrHLW+cYP+z/h+XpXPfPCxS4jeKQdVkUg/rXM4tHbCakVv\r\nIZ/9YScdh0pTGAuKkeXryKs6dpd7q0gFvEfLz80rcKPx7/Skk2U2ludB4Mic2d4WJ8sONns2Dn+l\r\ndKHaNsHmo9O09NNslt48kLyzHue5p0i5Oa6YqyOGbu7lyOXcKso1Y6u8ZyDVuC5DEBuDVpkNGkG4\r\noqJW460VYjMIUnmk2Y56inMv74/Wk6EisyhcDFGcHimE0ZoAnWUZ+bg+op0iwzptkVJF9GGf51Wz\r\nRuoGPXTNPVty2VqD6iNasbkUADoOgFVd1KGosGpM8m7joBUZwabk0c0xWEKAnpTSNoz6VKvWknXE\r\nDH2oAlSV0UAcj3opoICjPpRQAvG7P51EeWJoooAjPWm7ec5oopAPFKBRRQAYooopjFzS55oooEOF\r\nPYbrdh3FFFCEVpXCsAT2FFFFBR//2Q==", |
|||
"wechat": "", |
|||
"departmentid": 102, |
|||
"departmentname": "企管部", |
|||
"postid": 1120, |
|||
"postname": "新员工", |
|||
"tema": 0, |
|||
"temaname": "" |
|||
} |
|||
], |
|||
"titleDepartments": [ |
|||
{ |
|||
"departmentId": "309", |
|||
"id": "309", |
|||
"departmentKey": "GK00", |
|||
"departmentName": "山东恒信高科能源有限公司", |
|||
"parentId": "313", |
|||
"departmentNames": "山东恒信高科能源有限公司" |
|||
}, |
|||
{ |
|||
"departmentId": "102", |
|||
"id": "102", |
|||
"departmentKey": "GK03", |
|||
"departmentName": "企管部", |
|||
"parentId": "309", |
|||
"departmentNames": "企管部" |
|||
} |
|||
] |
|||
} |
|||
} |
|||
@ -0,0 +1,539 @@ |
|||
/* Logo 字体 */ |
|||
@font-face { |
|||
font-family: "iconfont logo"; |
|||
src: url('https://at.alicdn.com/t/font_985780_km7mi63cihi.eot?t=1545807318834'); |
|||
src: url('https://at.alicdn.com/t/font_985780_km7mi63cihi.eot?t=1545807318834#iefix') format('embedded-opentype'), |
|||
url('https://at.alicdn.com/t/font_985780_km7mi63cihi.woff?t=1545807318834') format('woff'), |
|||
url('https://at.alicdn.com/t/font_985780_km7mi63cihi.ttf?t=1545807318834') format('truetype'), |
|||
url('https://at.alicdn.com/t/font_985780_km7mi63cihi.svg?t=1545807318834#iconfont') format('svg'); |
|||
} |
|||
|
|||
.logo { |
|||
font-family: "iconfont logo"; |
|||
font-size: 160px; |
|||
font-style: normal; |
|||
-webkit-font-smoothing: antialiased; |
|||
-moz-osx-font-smoothing: grayscale; |
|||
} |
|||
|
|||
/* tabs */ |
|||
.nav-tabs { |
|||
position: relative; |
|||
} |
|||
|
|||
.nav-tabs .nav-more { |
|||
position: absolute; |
|||
right: 0; |
|||
bottom: 0; |
|||
height: 42px; |
|||
line-height: 42px; |
|||
color: #666; |
|||
} |
|||
|
|||
#tabs { |
|||
border-bottom: 1px solid #eee; |
|||
} |
|||
|
|||
#tabs li { |
|||
cursor: pointer; |
|||
width: 100px; |
|||
height: 40px; |
|||
line-height: 40px; |
|||
text-align: center; |
|||
font-size: 16px; |
|||
border-bottom: 2px solid transparent; |
|||
position: relative; |
|||
z-index: 1; |
|||
margin-bottom: -1px; |
|||
color: #666; |
|||
} |
|||
|
|||
|
|||
#tabs .active { |
|||
border-bottom-color: #f00; |
|||
color: #222; |
|||
} |
|||
|
|||
.tab-container .content { |
|||
display: none; |
|||
} |
|||
|
|||
/* 页面布局 */ |
|||
.main { |
|||
padding: 30px 100px; |
|||
width: 960px; |
|||
margin: 0 auto; |
|||
} |
|||
|
|||
.main .logo { |
|||
color: #333; |
|||
text-align: left; |
|||
margin-bottom: 30px; |
|||
line-height: 1; |
|||
height: 110px; |
|||
margin-top: -50px; |
|||
overflow: hidden; |
|||
*zoom: 1; |
|||
} |
|||
|
|||
.main .logo a { |
|||
font-size: 160px; |
|||
color: #333; |
|||
} |
|||
|
|||
.helps { |
|||
margin-top: 40px; |
|||
} |
|||
|
|||
.helps pre { |
|||
padding: 20px; |
|||
margin: 10px 0; |
|||
border: solid 1px #e7e1cd; |
|||
background-color: #fffdef; |
|||
overflow: auto; |
|||
} |
|||
|
|||
.icon_lists { |
|||
width: 100% !important; |
|||
overflow: hidden; |
|||
*zoom: 1; |
|||
} |
|||
|
|||
.icon_lists li { |
|||
width: 100px; |
|||
margin-bottom: 10px; |
|||
margin-right: 20px; |
|||
text-align: center; |
|||
list-style: none !important; |
|||
cursor: default; |
|||
} |
|||
|
|||
.icon_lists li .code-name { |
|||
line-height: 1.2; |
|||
} |
|||
|
|||
.icon_lists .icon { |
|||
display: block; |
|||
height: 100px; |
|||
line-height: 100px; |
|||
font-size: 42px; |
|||
margin: 10px auto; |
|||
color: #333; |
|||
-webkit-transition: font-size 0.25s linear, width 0.25s linear; |
|||
-moz-transition: font-size 0.25s linear, width 0.25s linear; |
|||
transition: font-size 0.25s linear, width 0.25s linear; |
|||
} |
|||
|
|||
.icon_lists .icon:hover { |
|||
font-size: 100px; |
|||
} |
|||
|
|||
.icon_lists .svg-icon { |
|||
/* 通过设置 font-size 来改变图标大小 */ |
|||
width: 1em; |
|||
/* 图标和文字相邻时,垂直对齐 */ |
|||
vertical-align: -0.15em; |
|||
/* 通过设置 color 来改变 SVG 的颜色/fill */ |
|||
fill: currentColor; |
|||
/* path 和 stroke 溢出 viewBox 部分在 IE 下会显示 |
|||
normalize.css 中也包含这行 */ |
|||
overflow: hidden; |
|||
} |
|||
|
|||
.icon_lists li .name, |
|||
.icon_lists li .code-name { |
|||
color: #666; |
|||
} |
|||
|
|||
/* markdown 样式 */ |
|||
.markdown { |
|||
color: #666; |
|||
font-size: 14px; |
|||
line-height: 1.8; |
|||
} |
|||
|
|||
.highlight { |
|||
line-height: 1.5; |
|||
} |
|||
|
|||
.markdown img { |
|||
vertical-align: middle; |
|||
max-width: 100%; |
|||
} |
|||
|
|||
.markdown h1 { |
|||
color: #404040; |
|||
font-weight: 500; |
|||
line-height: 40px; |
|||
margin-bottom: 24px; |
|||
} |
|||
|
|||
.markdown h2, |
|||
.markdown h3, |
|||
.markdown h4, |
|||
.markdown h5, |
|||
.markdown h6 { |
|||
color: #404040; |
|||
margin: 1.6em 0 0.6em 0; |
|||
font-weight: 500; |
|||
clear: both; |
|||
} |
|||
|
|||
.markdown h1 { |
|||
font-size: 28px; |
|||
} |
|||
|
|||
.markdown h2 { |
|||
font-size: 22px; |
|||
} |
|||
|
|||
.markdown h3 { |
|||
font-size: 16px; |
|||
} |
|||
|
|||
.markdown h4 { |
|||
font-size: 14px; |
|||
} |
|||
|
|||
.markdown h5 { |
|||
font-size: 12px; |
|||
} |
|||
|
|||
.markdown h6 { |
|||
font-size: 12px; |
|||
} |
|||
|
|||
.markdown hr { |
|||
height: 1px; |
|||
border: 0; |
|||
background: #e9e9e9; |
|||
margin: 16px 0; |
|||
clear: both; |
|||
} |
|||
|
|||
.markdown p { |
|||
margin: 1em 0; |
|||
} |
|||
|
|||
.markdown>p, |
|||
.markdown>blockquote, |
|||
.markdown>.highlight, |
|||
.markdown>ol, |
|||
.markdown>ul { |
|||
width: 80%; |
|||
} |
|||
|
|||
.markdown ul>li { |
|||
list-style: circle; |
|||
} |
|||
|
|||
.markdown>ul li, |
|||
.markdown blockquote ul>li { |
|||
margin-left: 20px; |
|||
padding-left: 4px; |
|||
} |
|||
|
|||
.markdown>ul li p, |
|||
.markdown>ol li p { |
|||
margin: 0.6em 0; |
|||
} |
|||
|
|||
.markdown ol>li { |
|||
list-style: decimal; |
|||
} |
|||
|
|||
.markdown>ol li, |
|||
.markdown blockquote ol>li { |
|||
margin-left: 20px; |
|||
padding-left: 4px; |
|||
} |
|||
|
|||
.markdown code { |
|||
margin: 0 3px; |
|||
padding: 0 5px; |
|||
background: #eee; |
|||
border-radius: 3px; |
|||
} |
|||
|
|||
.markdown strong, |
|||
.markdown b { |
|||
font-weight: 600; |
|||
} |
|||
|
|||
.markdown>table { |
|||
border-collapse: collapse; |
|||
border-spacing: 0px; |
|||
empty-cells: show; |
|||
border: 1px solid #e9e9e9; |
|||
width: 95%; |
|||
margin-bottom: 24px; |
|||
} |
|||
|
|||
.markdown>table th { |
|||
white-space: nowrap; |
|||
color: #333; |
|||
font-weight: 600; |
|||
} |
|||
|
|||
.markdown>table th, |
|||
.markdown>table td { |
|||
border: 1px solid #e9e9e9; |
|||
padding: 8px 16px; |
|||
text-align: left; |
|||
} |
|||
|
|||
.markdown>table th { |
|||
background: #F7F7F7; |
|||
} |
|||
|
|||
.markdown blockquote { |
|||
font-size: 90%; |
|||
color: #999; |
|||
border-left: 4px solid #e9e9e9; |
|||
padding-left: 0.8em; |
|||
margin: 1em 0; |
|||
} |
|||
|
|||
.markdown blockquote p { |
|||
margin: 0; |
|||
} |
|||
|
|||
.markdown .anchor { |
|||
opacity: 0; |
|||
transition: opacity 0.3s ease; |
|||
margin-left: 8px; |
|||
} |
|||
|
|||
.markdown .waiting { |
|||
color: #ccc; |
|||
} |
|||
|
|||
.markdown h1:hover .anchor, |
|||
.markdown h2:hover .anchor, |
|||
.markdown h3:hover .anchor, |
|||
.markdown h4:hover .anchor, |
|||
.markdown h5:hover .anchor, |
|||
.markdown h6:hover .anchor { |
|||
opacity: 1; |
|||
display: inline-block; |
|||
} |
|||
|
|||
.markdown>br, |
|||
.markdown>p>br { |
|||
clear: both; |
|||
} |
|||
|
|||
|
|||
.hljs { |
|||
display: block; |
|||
background: white; |
|||
padding: 0.5em; |
|||
color: #333333; |
|||
overflow-x: auto; |
|||
} |
|||
|
|||
.hljs-comment, |
|||
.hljs-meta { |
|||
color: #969896; |
|||
} |
|||
|
|||
.hljs-string, |
|||
.hljs-variable, |
|||
.hljs-template-variable, |
|||
.hljs-strong, |
|||
.hljs-emphasis, |
|||
.hljs-quote { |
|||
color: #df5000; |
|||
} |
|||
|
|||
.hljs-keyword, |
|||
.hljs-selector-tag, |
|||
.hljs-type { |
|||
color: #a71d5d; |
|||
} |
|||
|
|||
.hljs-literal, |
|||
.hljs-symbol, |
|||
.hljs-bullet, |
|||
.hljs-attribute { |
|||
color: #0086b3; |
|||
} |
|||
|
|||
.hljs-section, |
|||
.hljs-name { |
|||
color: #63a35c; |
|||
} |
|||
|
|||
.hljs-tag { |
|||
color: #333333; |
|||
} |
|||
|
|||
.hljs-title, |
|||
.hljs-attr, |
|||
.hljs-selector-id, |
|||
.hljs-selector-class, |
|||
.hljs-selector-attr, |
|||
.hljs-selector-pseudo { |
|||
color: #795da3; |
|||
} |
|||
|
|||
.hljs-addition { |
|||
color: #55a532; |
|||
background-color: #eaffea; |
|||
} |
|||
|
|||
.hljs-deletion { |
|||
color: #bd2c00; |
|||
background-color: #ffecec; |
|||
} |
|||
|
|||
.hljs-link { |
|||
text-decoration: underline; |
|||
} |
|||
|
|||
/* 代码高亮 */ |
|||
/* PrismJS 1.15.0 |
|||
https://prismjs.com/download.html#themes=prism&languages=markup+css+clike+javascript */ |
|||
/** |
|||
* prism.js default theme for JavaScript, CSS and HTML |
|||
* Based on dabblet (http://dabblet.com) |
|||
* @author Lea Verou |
|||
*/ |
|||
code[class*="language-"], |
|||
pre[class*="language-"] { |
|||
color: black; |
|||
background: none; |
|||
text-shadow: 0 1px white; |
|||
font-family: Consolas, Monaco, 'Andale Mono', 'Ubuntu Mono', monospace; |
|||
text-align: left; |
|||
white-space: pre; |
|||
word-spacing: normal; |
|||
word-break: normal; |
|||
word-wrap: normal; |
|||
line-height: 1.5; |
|||
|
|||
-moz-tab-size: 4; |
|||
-o-tab-size: 4; |
|||
tab-size: 4; |
|||
|
|||
-webkit-hyphens: none; |
|||
-moz-hyphens: none; |
|||
-ms-hyphens: none; |
|||
hyphens: none; |
|||
} |
|||
|
|||
pre[class*="language-"]::-moz-selection, |
|||
pre[class*="language-"] ::-moz-selection, |
|||
code[class*="language-"]::-moz-selection, |
|||
code[class*="language-"] ::-moz-selection { |
|||
text-shadow: none; |
|||
background: #b3d4fc; |
|||
} |
|||
|
|||
pre[class*="language-"]::selection, |
|||
pre[class*="language-"] ::selection, |
|||
code[class*="language-"]::selection, |
|||
code[class*="language-"] ::selection { |
|||
text-shadow: none; |
|||
background: #b3d4fc; |
|||
} |
|||
|
|||
@media print { |
|||
|
|||
code[class*="language-"], |
|||
pre[class*="language-"] { |
|||
text-shadow: none; |
|||
} |
|||
} |
|||
|
|||
/* Code blocks */ |
|||
pre[class*="language-"] { |
|||
padding: 1em; |
|||
margin: .5em 0; |
|||
overflow: auto; |
|||
} |
|||
|
|||
:not(pre)>code[class*="language-"], |
|||
pre[class*="language-"] { |
|||
background: #f5f2f0; |
|||
} |
|||
|
|||
/* Inline code */ |
|||
:not(pre)>code[class*="language-"] { |
|||
padding: .1em; |
|||
border-radius: .3em; |
|||
white-space: normal; |
|||
} |
|||
|
|||
.token.comment, |
|||
.token.prolog, |
|||
.token.doctype, |
|||
.token.cdata { |
|||
color: slategray; |
|||
} |
|||
|
|||
.token.punctuation { |
|||
color: #999; |
|||
} |
|||
|
|||
.namespace { |
|||
opacity: .7; |
|||
} |
|||
|
|||
.token.property, |
|||
.token.tag, |
|||
.token.boolean, |
|||
.token.number, |
|||
.token.constant, |
|||
.token.symbol, |
|||
.token.deleted { |
|||
color: #905; |
|||
} |
|||
|
|||
.token.selector, |
|||
.token.attr-name, |
|||
.token.string, |
|||
.token.char, |
|||
.token.builtin, |
|||
.token.inserted { |
|||
color: #690; |
|||
} |
|||
|
|||
.token.operator, |
|||
.token.entity, |
|||
.token.url, |
|||
.language-css .token.string, |
|||
.style .token.string { |
|||
color: #9a6e3a; |
|||
background: hsla(0, 0%, 100%, .5); |
|||
} |
|||
|
|||
.token.atrule, |
|||
.token.attr-value, |
|||
.token.keyword { |
|||
color: #07a; |
|||
} |
|||
|
|||
.token.function, |
|||
.token.class-name { |
|||
color: #DD4A68; |
|||
} |
|||
|
|||
.token.regex, |
|||
.token.important, |
|||
.token.variable { |
|||
color: #e90; |
|||
} |
|||
|
|||
.token.important, |
|||
.token.bold { |
|||
font-weight: bold; |
|||
} |
|||
|
|||
.token.italic { |
|||
font-style: italic; |
|||
} |
|||
|
|||
.token.entity { |
|||
cursor: help; |
|||
} |
|||
@ -0,0 +1,323 @@ |
|||
@font-face { |
|||
font-family: "iconfont"; /* Project id 2578019 */ |
|||
src: url('iconfont.woff2?t=1683511822799') format('woff2'), |
|||
url('iconfont.woff?t=1683511822799') format('woff'), |
|||
url('iconfont.ttf?t=1683511822799') format('truetype'); |
|||
} |
|||
|
|||
.iconfont,[class*="icon-"] { |
|||
font-family: "iconfont" !important; |
|||
font-size: 16px; |
|||
font-style: normal; |
|||
-webkit-font-smoothing: antialiased; |
|||
-moz-osx-font-smoothing: grayscale; |
|||
} |
|||
|
|||
.icon-mouse:before { |
|||
content: "\e60b"; |
|||
} |
|||
|
|||
.icon-hand:before { |
|||
content: "\e6cf"; |
|||
} |
|||
|
|||
.icon-lock-open:before { |
|||
content: "\e9d6"; |
|||
} |
|||
|
|||
.icon-lock:before { |
|||
content: "\e669"; |
|||
} |
|||
|
|||
.icon-eye-close:before { |
|||
content: "\ebcd"; |
|||
} |
|||
|
|||
.icon-border:before { |
|||
content: "\e673"; |
|||
} |
|||
|
|||
.icon-text2:before { |
|||
content: "\e607"; |
|||
} |
|||
|
|||
.icon-stext:before { |
|||
content: "\e801"; |
|||
} |
|||
|
|||
.icon-apply:before { |
|||
content: "\e606"; |
|||
} |
|||
|
|||
.icon-work:before { |
|||
content: "\e68c"; |
|||
} |
|||
|
|||
.icon-todo:before { |
|||
content: "\ebb1"; |
|||
} |
|||
|
|||
.icon-applyed:before { |
|||
content: "\e633"; |
|||
} |
|||
|
|||
.icon-done:before { |
|||
content: "\e60d"; |
|||
} |
|||
|
|||
.icon-button:before { |
|||
content: "\e690"; |
|||
} |
|||
|
|||
.icon-design:before { |
|||
content: "\e62c"; |
|||
} |
|||
|
|||
.icon-data-source:before { |
|||
content: "\e617"; |
|||
} |
|||
|
|||
.icon-sp:before { |
|||
content: "\e626"; |
|||
} |
|||
|
|||
.icon-cs:before { |
|||
content: "\e605"; |
|||
} |
|||
|
|||
.icon-branch:before { |
|||
content: "\e993"; |
|||
} |
|||
|
|||
.icon-doc:before { |
|||
content: "\e7d0"; |
|||
} |
|||
|
|||
.icon-sys:before { |
|||
content: "\e634"; |
|||
} |
|||
|
|||
.icon-creat:before { |
|||
content: "\e661"; |
|||
} |
|||
|
|||
.icon-data:before { |
|||
content: "\e602"; |
|||
} |
|||
|
|||
.icon-menu:before { |
|||
content: "\e62f"; |
|||
} |
|||
|
|||
.icon-user:before { |
|||
content: "\e66b"; |
|||
} |
|||
|
|||
.icon-form:before { |
|||
content: "\e625"; |
|||
} |
|||
|
|||
.icon-list:before { |
|||
content: "\ec6b"; |
|||
} |
|||
|
|||
.icon-log:before { |
|||
content: "\e604"; |
|||
} |
|||
|
|||
.icon-dict:before { |
|||
content: "\e668"; |
|||
} |
|||
|
|||
.icon-post:before { |
|||
content: "\e658"; |
|||
} |
|||
|
|||
.icon-role:before { |
|||
content: "\e618"; |
|||
} |
|||
|
|||
.icon-log2:before { |
|||
content: "\ea45"; |
|||
} |
|||
|
|||
.icon-tool:before { |
|||
content: "\e628"; |
|||
} |
|||
|
|||
.icon-bar:before { |
|||
content: "\e6ce"; |
|||
} |
|||
|
|||
.icon-pie:before { |
|||
content: "\e902"; |
|||
} |
|||
|
|||
.icon-line:before { |
|||
content: "\ec66"; |
|||
} |
|||
|
|||
.icon-tree2:before { |
|||
content: "\e892"; |
|||
} |
|||
|
|||
.icon-flex:before { |
|||
content: "\e608"; |
|||
} |
|||
|
|||
.icon-div:before { |
|||
content: "\e60f"; |
|||
} |
|||
|
|||
.icon-tree:before { |
|||
content: "\ebb3"; |
|||
} |
|||
|
|||
.icon-time:before { |
|||
content: "\e600"; |
|||
} |
|||
|
|||
.icon-tinymce:before { |
|||
content: "\e66f"; |
|||
} |
|||
|
|||
.icon-rate:before { |
|||
content: "\e69d"; |
|||
} |
|||
|
|||
.icon-slider:before { |
|||
content: "\e627"; |
|||
} |
|||
|
|||
.icon-divider:before { |
|||
content: "\e638"; |
|||
} |
|||
|
|||
.icon-color:before { |
|||
content: "\ee22"; |
|||
} |
|||
|
|||
.icon-import:before { |
|||
content: "\e616"; |
|||
} |
|||
|
|||
.icon-image:before { |
|||
content: "\e63e"; |
|||
} |
|||
|
|||
.icon-card:before { |
|||
content: "\e622"; |
|||
} |
|||
|
|||
.icon-help:before { |
|||
content: "\e61c"; |
|||
} |
|||
|
|||
.icon-table:before { |
|||
content: "\e6a9"; |
|||
} |
|||
|
|||
.icon-component:before { |
|||
content: "\e620"; |
|||
} |
|||
|
|||
.icon-link:before { |
|||
content: "\e623"; |
|||
} |
|||
|
|||
.icon-cascader:before { |
|||
content: "\e624"; |
|||
} |
|||
|
|||
.icon-number:before { |
|||
content: "\e647"; |
|||
} |
|||
|
|||
.icon-title:before { |
|||
content: "\e61d"; |
|||
} |
|||
|
|||
.icon-tabs:before { |
|||
content: "\e6d4"; |
|||
} |
|||
|
|||
.icon-close:before { |
|||
content: "\e615"; |
|||
} |
|||
|
|||
.icon-plus:before { |
|||
content: "\e603"; |
|||
} |
|||
|
|||
.icon-arrow:before { |
|||
content: "\e61b"; |
|||
} |
|||
|
|||
.icon-check:before { |
|||
content: "\e6c1"; |
|||
} |
|||
|
|||
.icon-vue:before { |
|||
content: "\e69a"; |
|||
} |
|||
|
|||
.icon-switch:before { |
|||
content: "\e646"; |
|||
} |
|||
|
|||
.icon-save:before { |
|||
content: "\e61f"; |
|||
} |
|||
|
|||
.icon-move:before { |
|||
content: "\e696"; |
|||
} |
|||
|
|||
.icon-clone:before { |
|||
content: "\e692"; |
|||
} |
|||
|
|||
.icon-del:before { |
|||
content: "\e67d"; |
|||
} |
|||
|
|||
.icon-eye:before { |
|||
content: "\e601"; |
|||
} |
|||
|
|||
.icon-json:before { |
|||
content: "\e60e"; |
|||
} |
|||
|
|||
.icon-grid:before { |
|||
content: "\e652"; |
|||
} |
|||
|
|||
.icon-text:before { |
|||
content: "\e621"; |
|||
} |
|||
|
|||
.icon-date:before { |
|||
content: "\e64c"; |
|||
} |
|||
|
|||
.icon-input:before { |
|||
content: "\e629"; |
|||
} |
|||
|
|||
.icon-checkbox:before { |
|||
content: "\e61e"; |
|||
} |
|||
|
|||
.icon-textarea:before { |
|||
content: "\e60a"; |
|||
} |
|||
|
|||
.icon-radio:before { |
|||
content: "\e6b1"; |
|||
} |
|||
|
|||
.icon-select:before { |
|||
content: "\e64d"; |
|||
} |
|||
|
|||
@ -0,0 +1,548 @@ |
|||
{ |
|||
"id": "2578019", |
|||
"name": "formDemo", |
|||
"font_family": "iconfont", |
|||
"css_prefix_text": "icon-", |
|||
"description": "", |
|||
"glyphs": [ |
|||
{ |
|||
"icon_id": "13644674", |
|||
"name": "箭头", |
|||
"font_class": "mouse", |
|||
"unicode": "e60b", |
|||
"unicode_decimal": 58891 |
|||
}, |
|||
{ |
|||
"icon_id": "18372381", |
|||
"name": "手,手势", |
|||
"font_class": "hand", |
|||
"unicode": "e6cf", |
|||
"unicode_decimal": 59087 |
|||
}, |
|||
{ |
|||
"icon_id": "7577422", |
|||
"name": "24gl-unlock4", |
|||
"font_class": "lock-open", |
|||
"unicode": "e9d6", |
|||
"unicode_decimal": 59862 |
|||
}, |
|||
{ |
|||
"icon_id": "658044", |
|||
"name": "解锁", |
|||
"font_class": "lock", |
|||
"unicode": "e669", |
|||
"unicode_decimal": 58985 |
|||
}, |
|||
{ |
|||
"icon_id": "5388071", |
|||
"name": "眼睛_隐藏_o", |
|||
"font_class": "eye-close", |
|||
"unicode": "ebcd", |
|||
"unicode_decimal": 60365 |
|||
}, |
|||
{ |
|||
"icon_id": "3294800", |
|||
"name": "设置边框", |
|||
"font_class": "border", |
|||
"unicode": "e673", |
|||
"unicode_decimal": 58995 |
|||
}, |
|||
{ |
|||
"icon_id": "11179879", |
|||
"name": "文字边框", |
|||
"font_class": "text2", |
|||
"unicode": "e607", |
|||
"unicode_decimal": 58887 |
|||
}, |
|||
{ |
|||
"icon_id": "21529713", |
|||
"name": "m-滚动文字", |
|||
"font_class": "stext", |
|||
"unicode": "e801", |
|||
"unicode_decimal": 59393 |
|||
}, |
|||
{ |
|||
"icon_id": "33172809", |
|||
"name": "发起的", |
|||
"font_class": "apply", |
|||
"unicode": "e606", |
|||
"unicode_decimal": 58886 |
|||
}, |
|||
{ |
|||
"icon_id": "1206679", |
|||
"name": "工作台", |
|||
"font_class": "work", |
|||
"unicode": "e68c", |
|||
"unicode_decimal": 59020 |
|||
}, |
|||
{ |
|||
"icon_id": "5388003", |
|||
"name": "待办任务_o", |
|||
"font_class": "todo", |
|||
"unicode": "ebb1", |
|||
"unicode_decimal": 60337 |
|||
}, |
|||
{ |
|||
"icon_id": "7158013", |
|||
"name": "我发起的", |
|||
"font_class": "applyed", |
|||
"unicode": "e633", |
|||
"unicode_decimal": 58931 |
|||
}, |
|||
{ |
|||
"icon_id": "20438918", |
|||
"name": "我的已办", |
|||
"font_class": "done", |
|||
"unicode": "e60d", |
|||
"unicode_decimal": 58893 |
|||
}, |
|||
{ |
|||
"icon_id": "1766470", |
|||
"name": "符号-按钮", |
|||
"font_class": "button", |
|||
"unicode": "e690", |
|||
"unicode_decimal": 59024 |
|||
}, |
|||
{ |
|||
"icon_id": "1150901", |
|||
"name": "设计", |
|||
"font_class": "design", |
|||
"unicode": "e62c", |
|||
"unicode_decimal": 58924 |
|||
}, |
|||
{ |
|||
"icon_id": "12771491", |
|||
"name": "数据源管理", |
|||
"font_class": "data-source", |
|||
"unicode": "e617", |
|||
"unicode_decimal": 58903 |
|||
}, |
|||
{ |
|||
"icon_id": "6321263", |
|||
"name": "审批", |
|||
"font_class": "sp", |
|||
"unicode": "e626", |
|||
"unicode_decimal": 58918 |
|||
}, |
|||
{ |
|||
"icon_id": "12911863", |
|||
"name": "抄送", |
|||
"font_class": "cs", |
|||
"unicode": "e605", |
|||
"unicode_decimal": 58885 |
|||
}, |
|||
{ |
|||
"icon_id": "18170243", |
|||
"name": "代码,分支,分流,分叉", |
|||
"font_class": "branch", |
|||
"unicode": "e993", |
|||
"unicode_decimal": 59795 |
|||
}, |
|||
{ |
|||
"icon_id": "5371107", |
|||
"name": "文档", |
|||
"font_class": "doc", |
|||
"unicode": "e7d0", |
|||
"unicode_decimal": 59344 |
|||
}, |
|||
{ |
|||
"icon_id": "12959365", |
|||
"name": "系统管理", |
|||
"font_class": "sys", |
|||
"unicode": "e634", |
|||
"unicode_decimal": 58932 |
|||
}, |
|||
{ |
|||
"icon_id": "24007617", |
|||
"name": "创建图表", |
|||
"font_class": "creat", |
|||
"unicode": "e661", |
|||
"unicode_decimal": 58977 |
|||
}, |
|||
{ |
|||
"icon_id": "24069228", |
|||
"name": "可视化", |
|||
"font_class": "data", |
|||
"unicode": "e602", |
|||
"unicode_decimal": 58882 |
|||
}, |
|||
{ |
|||
"icon_id": "339044", |
|||
"name": "菜单", |
|||
"font_class": "menu", |
|||
"unicode": "e62f", |
|||
"unicode_decimal": 58927 |
|||
}, |
|||
{ |
|||
"icon_id": "397442", |
|||
"name": "icon-user", |
|||
"font_class": "user", |
|||
"unicode": "e66b", |
|||
"unicode_decimal": 58987 |
|||
}, |
|||
{ |
|||
"icon_id": "4329375", |
|||
"name": "表单管理", |
|||
"font_class": "form", |
|||
"unicode": "e625", |
|||
"unicode_decimal": 58917 |
|||
}, |
|||
{ |
|||
"icon_id": "5961366", |
|||
"name": "列表", |
|||
"font_class": "list", |
|||
"unicode": "ec6b", |
|||
"unicode_decimal": 60523 |
|||
}, |
|||
{ |
|||
"icon_id": "7474663", |
|||
"name": "操作日志", |
|||
"font_class": "log", |
|||
"unicode": "e604", |
|||
"unicode_decimal": 58884 |
|||
}, |
|||
{ |
|||
"icon_id": "8605754", |
|||
"name": "字典管理", |
|||
"font_class": "dict", |
|||
"unicode": "e668", |
|||
"unicode_decimal": 58984 |
|||
}, |
|||
{ |
|||
"icon_id": "8949009", |
|||
"name": "岗位", |
|||
"font_class": "post", |
|||
"unicode": "e658", |
|||
"unicode_decimal": 58968 |
|||
}, |
|||
{ |
|||
"icon_id": "10213489", |
|||
"name": "角色管理", |
|||
"font_class": "role", |
|||
"unicode": "e618", |
|||
"unicode_decimal": 58904 |
|||
}, |
|||
{ |
|||
"icon_id": "26876046", |
|||
"name": "管理登录日志", |
|||
"font_class": "log2", |
|||
"unicode": "ea45", |
|||
"unicode_decimal": 59973 |
|||
}, |
|||
{ |
|||
"icon_id": "34047671", |
|||
"name": "系统工具", |
|||
"font_class": "tool", |
|||
"unicode": "e628", |
|||
"unicode_decimal": 58920 |
|||
}, |
|||
{ |
|||
"icon_id": "1057386", |
|||
"name": "柱状图", |
|||
"font_class": "bar", |
|||
"unicode": "e6ce", |
|||
"unicode_decimal": 59086 |
|||
}, |
|||
{ |
|||
"icon_id": "1727592", |
|||
"name": "407饼图", |
|||
"font_class": "pie", |
|||
"unicode": "e902", |
|||
"unicode_decimal": 59650 |
|||
}, |
|||
{ |
|||
"icon_id": "5961328", |
|||
"name": "折线图", |
|||
"font_class": "line", |
|||
"unicode": "ec66", |
|||
"unicode_decimal": 60518 |
|||
}, |
|||
{ |
|||
"icon_id": "11307823", |
|||
"name": "Directory tree", |
|||
"font_class": "tree2", |
|||
"unicode": "e892", |
|||
"unicode_decimal": 59538 |
|||
}, |
|||
{ |
|||
"icon_id": "20360961", |
|||
"name": "流式布局", |
|||
"font_class": "flex", |
|||
"unicode": "e608", |
|||
"unicode_decimal": 58888 |
|||
}, |
|||
{ |
|||
"icon_id": "12225038", |
|||
"name": "divide", |
|||
"font_class": "div", |
|||
"unicode": "e60f", |
|||
"unicode_decimal": 58895 |
|||
}, |
|||
{ |
|||
"icon_id": "5388006", |
|||
"name": "树状图_o", |
|||
"font_class": "tree", |
|||
"unicode": "ebb3", |
|||
"unicode_decimal": 60339 |
|||
}, |
|||
{ |
|||
"icon_id": "77110", |
|||
"name": "时间", |
|||
"font_class": "time", |
|||
"unicode": "e600", |
|||
"unicode_decimal": 58880 |
|||
}, |
|||
{ |
|||
"icon_id": "2471358", |
|||
"name": "富文本框", |
|||
"font_class": "tinymce", |
|||
"unicode": "e66f", |
|||
"unicode_decimal": 58991 |
|||
}, |
|||
{ |
|||
"icon_id": "8687733", |
|||
"name": "评分", |
|||
"font_class": "rate", |
|||
"unicode": "e69d", |
|||
"unicode_decimal": 59037 |
|||
}, |
|||
{ |
|||
"icon_id": "11121364", |
|||
"name": "滑块", |
|||
"font_class": "slider", |
|||
"unicode": "e627", |
|||
"unicode_decimal": 58919 |
|||
}, |
|||
{ |
|||
"icon_id": "11121454", |
|||
"name": "分割线", |
|||
"font_class": "divider", |
|||
"unicode": "e638", |
|||
"unicode_decimal": 58936 |
|||
}, |
|||
{ |
|||
"icon_id": "22385724", |
|||
"name": "颜色库", |
|||
"font_class": "color", |
|||
"unicode": "ee22", |
|||
"unicode_decimal": 60962 |
|||
}, |
|||
{ |
|||
"icon_id": "6244963", |
|||
"name": "导入", |
|||
"font_class": "import", |
|||
"unicode": "e616", |
|||
"unicode_decimal": 58902 |
|||
}, |
|||
{ |
|||
"icon_id": "145454", |
|||
"name": "图片", |
|||
"font_class": "image", |
|||
"unicode": "e63e", |
|||
"unicode_decimal": 58942 |
|||
}, |
|||
{ |
|||
"icon_id": "646357", |
|||
"name": "卡片", |
|||
"font_class": "card", |
|||
"unicode": "e622", |
|||
"unicode_decimal": 58914 |
|||
}, |
|||
{ |
|||
"icon_id": "737945", |
|||
"name": "help", |
|||
"font_class": "help", |
|||
"unicode": "e61c", |
|||
"unicode_decimal": 58908 |
|||
}, |
|||
{ |
|||
"icon_id": "1467412", |
|||
"name": "table", |
|||
"font_class": "table", |
|||
"unicode": "e6a9", |
|||
"unicode_decimal": 59049 |
|||
}, |
|||
{ |
|||
"icon_id": "13253956", |
|||
"name": "component", |
|||
"font_class": "component", |
|||
"unicode": "e620", |
|||
"unicode_decimal": 58912 |
|||
}, |
|||
{ |
|||
"icon_id": "15196938", |
|||
"name": "11.符号-级联选择", |
|||
"font_class": "link", |
|||
"unicode": "e623", |
|||
"unicode_decimal": 58915 |
|||
}, |
|||
{ |
|||
"icon_id": "16880964", |
|||
"name": "cascader", |
|||
"font_class": "cascader", |
|||
"unicode": "e624", |
|||
"unicode_decimal": 58916 |
|||
}, |
|||
{ |
|||
"icon_id": "17374561", |
|||
"name": "计数器", |
|||
"font_class": "number", |
|||
"unicode": "e647", |
|||
"unicode_decimal": 58951 |
|||
}, |
|||
{ |
|||
"icon_id": "19657932", |
|||
"name": "标题", |
|||
"font_class": "title", |
|||
"unicode": "e61d", |
|||
"unicode_decimal": 58909 |
|||
}, |
|||
{ |
|||
"icon_id": "8017649", |
|||
"name": "bootstrap_tabs", |
|||
"font_class": "tabs", |
|||
"unicode": "e6d4", |
|||
"unicode_decimal": 59092 |
|||
}, |
|||
{ |
|||
"icon_id": "7826085", |
|||
"name": "close", |
|||
"font_class": "close", |
|||
"unicode": "e615", |
|||
"unicode_decimal": 58901 |
|||
}, |
|||
{ |
|||
"icon_id": "6756281", |
|||
"name": "加号", |
|||
"font_class": "plus", |
|||
"unicode": "e603", |
|||
"unicode_decimal": 58883 |
|||
}, |
|||
{ |
|||
"icon_id": "902007", |
|||
"name": "arrow on", |
|||
"font_class": "arrow", |
|||
"unicode": "e61b", |
|||
"unicode_decimal": 58907 |
|||
}, |
|||
{ |
|||
"icon_id": "1176866", |
|||
"name": "check", |
|||
"font_class": "check", |
|||
"unicode": "e6c1", |
|||
"unicode_decimal": 59073 |
|||
}, |
|||
{ |
|||
"icon_id": "13744032", |
|||
"name": "vue", |
|||
"font_class": "vue", |
|||
"unicode": "e69a", |
|||
"unicode_decimal": 59034 |
|||
}, |
|||
{ |
|||
"icon_id": "20905073", |
|||
"name": "开关", |
|||
"font_class": "switch", |
|||
"unicode": "e646", |
|||
"unicode_decimal": 58950 |
|||
}, |
|||
{ |
|||
"icon_id": "11493812", |
|||
"name": "保存", |
|||
"font_class": "save", |
|||
"unicode": "e61f", |
|||
"unicode_decimal": 58911 |
|||
}, |
|||
{ |
|||
"icon_id": "384286", |
|||
"name": "move", |
|||
"font_class": "move", |
|||
"unicode": "e696", |
|||
"unicode_decimal": 59030 |
|||
}, |
|||
{ |
|||
"icon_id": "836441", |
|||
"name": "clone", |
|||
"font_class": "clone", |
|||
"unicode": "e692", |
|||
"unicode_decimal": 59026 |
|||
}, |
|||
{ |
|||
"icon_id": "18367976", |
|||
"name": "删 除 (1)", |
|||
"font_class": "del", |
|||
"unicode": "e67d", |
|||
"unicode_decimal": 59005 |
|||
}, |
|||
{ |
|||
"icon_id": "925051", |
|||
"name": "eye", |
|||
"font_class": "eye", |
|||
"unicode": "e601", |
|||
"unicode_decimal": 58881 |
|||
}, |
|||
{ |
|||
"icon_id": "15214524", |
|||
"name": "json", |
|||
"font_class": "json", |
|||
"unicode": "e60e", |
|||
"unicode_decimal": 58894 |
|||
}, |
|||
{ |
|||
"icon_id": "350628", |
|||
"name": "GRID", |
|||
"font_class": "grid", |
|||
"unicode": "e652", |
|||
"unicode_decimal": 58962 |
|||
}, |
|||
{ |
|||
"icon_id": "4521248", |
|||
"name": "文字", |
|||
"font_class": "text", |
|||
"unicode": "e621", |
|||
"unicode_decimal": 58913 |
|||
}, |
|||
{ |
|||
"icon_id": "15167569", |
|||
"name": "date", |
|||
"font_class": "date", |
|||
"unicode": "e64c", |
|||
"unicode_decimal": 58956 |
|||
}, |
|||
{ |
|||
"icon_id": "16880988", |
|||
"name": "input", |
|||
"font_class": "input", |
|||
"unicode": "e629", |
|||
"unicode_decimal": 58921 |
|||
}, |
|||
{ |
|||
"icon_id": "18536117", |
|||
"name": "check_box-24px", |
|||
"font_class": "checkbox", |
|||
"unicode": "e61e", |
|||
"unicode_decimal": 58910 |
|||
}, |
|||
{ |
|||
"icon_id": "18727247", |
|||
"name": "textarea", |
|||
"font_class": "textarea", |
|||
"unicode": "e60a", |
|||
"unicode_decimal": 58890 |
|||
}, |
|||
{ |
|||
"icon_id": "19457233", |
|||
"name": "radio", |
|||
"font_class": "radio", |
|||
"unicode": "e6b1", |
|||
"unicode_decimal": 59057 |
|||
}, |
|||
{ |
|||
"icon_id": "20962223", |
|||
"name": "下拉选择", |
|||
"font_class": "select", |
|||
"unicode": "e64d", |
|||
"unicode_decimal": 58957 |
|||
} |
|||
] |
|||
} |
|||
|
After Width: | Height: | Size: 768 B |
|
After Width: | Height: | Size: 944 B |
|
After Width: | Height: | Size: 209 B |
|
After Width: | Height: | Size: 1.2 KiB |
|
After Width: | Height: | Size: 245 B |
|
After Width: | Height: | Size: 266 B |
|
After Width: | Height: | Size: 6.0 KiB |
|
After Width: | Height: | Size: 5.3 KiB |
|
After Width: | Height: | Size: 5.4 KiB |
|
After Width: | Height: | Size: 6.2 KiB |
|
After Width: | Height: | Size: 5.6 KiB |
|
After Width: | Height: | Size: 1.1 KiB |
@ -0,0 +1,9 @@ |
|||
@charset "UTF-8"; |
|||
:root { |
|||
/*--el-border-base: 1px solid red*/ |
|||
--el-menu-item-height:44px; |
|||
--el-menu-base-level-padding:10px; |
|||
.el-header{ |
|||
--el-header-height:40px |
|||
} |
|||
} |
|||
@ -0,0 +1,125 @@ |
|||
.flow-container { |
|||
$background: #fff; |
|||
$borderColor: #eee; |
|||
overflow-x: auto; |
|||
.flow-main { min-width: min-content;display: flex;flex-direction: column;align-items: center;} |
|||
.flow-group {position: relative;background: $background;min-width: min-content} |
|||
.flow-item {border: 1px solid $borderColor;box-shadow: 0 0 6px rgba(102, 102, 102, 0.1);height: 80px;border-radius: 5px;overflow: hidden;display: flex;flex-direction: column;transition: all .3s;position: relative;z-index: 2;background: $background;width: 180px; |
|||
&:hover {border-color: rgb(85, 197, 192); |
|||
.title .close {visibility: visible;opacity: 1} |
|||
} |
|||
.title {background: rgb(85, 197, 192);color: #fff;display: flex;align-items: center;padding: 3px 10px;justify-content: space-between; |
|||
span {flex: 2} |
|||
.close {transform: scale(.8);visibility: hidden;transition: all .3s;opacity: 0;cursor: pointer} |
|||
i {font-size: 14px;margin-right: 3px} |
|||
&.bg-2 {background: rgb(255, 148, 62)} |
|||
&.bg-5 {background: none;color: #666} |
|||
&.bg-3 {background: rgb(50, 150, 250)} |
|||
} |
|||
.text {flex: 1;display: flex;align-items: center;padding: 5px 15px;cursor: pointer; |
|||
div {display: block;max-height: 38px;overflow: hidden;color: #666;text-overflow: ellipsis;display: -webkit-box;-webkit-box-orient: vertical;-webkit-line-clamp: 2;} |
|||
} |
|||
} |
|||
.flow-add {padding: 8px 0;position: relative;font-size: 12px; |
|||
i {width: 30px;height: 30px;border-radius: 50%;border: 1px solid #eee;box-shadow: 0 0 6px rgba(102, 102, 102, 0.1);display: flex;align-items: center;justify-content: center;margin: 0 auto;background: $background;position: relative;z-index: 2;cursor: pointer;transform: scale(.7); |
|||
} |
|||
&:before {content: '';display: block;width: 1px;left: 50%;background: #eee;position: absolute;top: 0;bottom: 0} |
|||
} |
|||
.flow-col {flex: 1;margin: 0 5px;display: flex;flex-direction: column;align-items: center;position: relative; |
|||
&:before {content: '';display: block;width: 1px;top: 0; bottom: 0;left: 50%;background: #eee;position: absolute} |
|||
} |
|||
.flow-row {display: flex;position: relative} |
|||
.flow-branch-btn {display: none} |
|||
.flow-branch {padding: 40px 0 1px;display: flex;flex-direction: column; |
|||
.flow-branch-btn {display: block;position: absolute;left: 50%;top: 0;transform: translateX(-50%);z-index: 2} |
|||
&:before {content: '';display: block;left: 0;right: 0;bottom: 46px;top: 12px;border: 1px solid #eee;position: absolute} |
|||
.flow-col { // 中线 |
|||
&:before {top: -28px} |
|||
&:after {content: '';display: none;width: 50%;top: 0;background: $borderColor;} |
|||
} |
|||
.mask-left, .mask-right {position: absolute;background: $background;top: -28px;display: block;bottom: -1px;width: calc(50% + 9px)} |
|||
.mask-left {left: -10px} |
|||
.mask-right {right: -10px} |
|||
} |
|||
.flow-end { |
|||
box-shadow: 0 2px 10px 0 rgb(145 145 153 / 50%);margin: 0 auto 30px;border-radius: 50%;position: relative;background: #bcbcc5;width: 50px;height: 50px;color: #fff;display: flex;align-items: center;justify-content: center; |
|||
} |
|||
&.type-show { |
|||
// 展示模式 |
|||
.flow-branch {padding-top: 28px} |
|||
.flow-item {margin-bottom: 20px} |
|||
.flow-branch:before {bottom: 0;top: 0} |
|||
.flow-end {position: relative;margin-top: 20px; |
|||
&:before {content: '';width: 1px;background: $borderColor;position: absolute;left: 50%;top: -20px;height: 20px;} |
|||
} |
|||
} |
|||
.tools {text-align: right} |
|||
} |
|||
.flow-node-down { |
|||
h3 {font-size: 14px;margin-bottom: 10px} |
|||
div {display: flex} |
|||
i {font-size: 14px;margin-right: 5px} |
|||
} |
|||
.flow-container.horizontal { |
|||
$borderColor: #eee; |
|||
.flow-main {flex-direction: row;justify-content: center} |
|||
.flow-group:first-child {overflow: hidden;} |
|||
.flow-col {flex-direction: row;margin: 8px 0; |
|||
&:before {width: auto; right: 0;height: 1px;left: -29px;top: 50%} |
|||
} |
|||
.flow-add {padding: 0 8px; |
|||
&:before {left: 0;top: 50%;height: 1px;width: 100%} |
|||
} |
|||
.flow-row {flex-direction: column} |
|||
.flow-branch {flex-direction: row;align-items: center;padding: 0; |
|||
.flow-branch-btn {position: static;transform: translateX(0);margin-right: 8px; |
|||
button {width: 50px;white-space: normal;padding: 15px 10px;overflow: hidden;line-height: 14px} |
|||
} |
|||
&:before {bottom: 0;top: 0;left: 30px;right: 45px} |
|||
.mask-left, .mask-right {left: -29px;right: -1px;width: auto; height: calc(50% + 8px)} |
|||
.mask-left {top: -8px;} |
|||
.mask-right {bottom: -9px;top: auto} |
|||
.flow-col { |
|||
|
|||
} |
|||
} |
|||
.flow-end {margin: 0} |
|||
&.type-show { |
|||
.flow-branch {margin-left: 20px} |
|||
.flow-item {margin-right: 20px;margin-bottom: 0} |
|||
.flow-branch .flow-col:before {left: -20px} |
|||
.flow-branch:before {left: -20px;right: -1px} |
|||
.flow-end {position: relative;margin-left: 20px; |
|||
&:before {content: '';height: 1px;background: $borderColor;position: absolute;top: 50%;left: -20px;width: 20px;} |
|||
} |
|||
} |
|||
} |
|||
// 侧栏drawer |
|||
.flow-modal-class { |
|||
$borderColor: #eee; |
|||
.el-drawer__header {border-bottom: 1px solid $borderColor;margin-bottom: 0;padding-bottom: 20px} |
|||
} |
|||
.flow-drawer { |
|||
.tip {margin-bottom: 20px;color: #999;line-height: 22px} |
|||
} |
|||
// 设计主页 |
|||
.design-flow-container {position: relative; |
|||
.tools {position: absolute;right: 20px;top: 13px;z-index: 2} |
|||
.tabs-info {display: flex;justify-content: center; |
|||
form {width: 680px;} |
|||
} |
|||
} |
|||
//工作台 |
|||
.task-apply { |
|||
.item {margin-bottom: 20px; |
|||
h3 {font-size: 14px;margin-bottom: 10px} |
|||
} |
|||
.list {display: flex;margin: 0 -8px; |
|||
> div {border: 1px solid #dcdfe6;padding: 16px;border-radius: 8px;height: 100px;margin: 0 8px;width: 24%;display: flex;align-items: center;font-weight: 700;transition: all .3s;cursor: pointer;box-shadow: 0 0 2px #ddd; |
|||
&:hover {border-color: #409eff} |
|||
} |
|||
.icon {font-size: 36px;width: 50px;height: 50px;display: flex;align-items: center;justify-content: center;border-radius: 4px;overflow: hidden;color: #fff;margin-right: 10px; |
|||
&:before {font-weight: 400} |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,296 @@ |
|||
@charset "UTF-8"; |
|||
// 左侧栏 |
|||
$mainColor: #66b1ff; |
|||
.design-container {margin: 10px !important;display: flex; background-color: #FFFFFF;} |
|||
.components-list {width: 250px;padding: 8px 0;overflow-y: auto;height: calc(100vh - 104px);position: relative;overflow-x: hidden; |
|||
.title {padding: 8px 12px;position: relative; |
|||
.template {position: absolute;right: 12px;top: 0;padding: 8px;cursor: pointer} |
|||
} |
|||
ul { position: relative;overflow: hidden;padding: 0 10px 10px;margin: 0; |
|||
li {font-size: 13px;display: flex;width: 48%;line-height: 28px;position: relative;float: left;left: 0;overflow: hidden;text-overflow: ellipsis;white-space: nowrap;margin: 0 1% 5px;color: #333;padding: 0 10px;border: 1px solid transparent;background: #f4f6fc;user-select: none;align-items: center; |
|||
i {margin-right: 5px} |
|||
&:hover { |
|||
color: $mainColor;border: 1px dashed $mainColor;cursor: move; |
|||
} |
|||
&.title {padding: 0 10px} |
|||
} |
|||
} |
|||
.content {margin-left: 12px} |
|||
} |
|||
.use-template {position: absolute;left: -100%;top: 0;width: 100%;bottom: 0;background: #fff;opacity: 0; |
|||
&.active {animation: templateAni .5s forwards;} |
|||
.close {position: absolute;right: 5px;top: 5px;font-size: 14px;cursor: pointer} |
|||
.list {padding: 30px 10px;overflow-y: auto; |
|||
.item {box-shadow: 0 0 4px rgba(0, 21, 41, 0.1);padding: 10px;text-align: center;transition: all .5s;cursor: pointer;margin-bottom: 10px; |
|||
img {display: block;width: 100%;margin: 0 auto 10px} |
|||
&:hover {box-shadow: 0 0 4px rgba(0, 21, 41, 0.25);} |
|||
} |
|||
} |
|||
.no-date {text-align: center;padding-top: 20px;color: #999} |
|||
} |
|||
@keyframes templateAni { |
|||
100% {opacity: 1;left: 0} |
|||
} |
|||
// 框架 |
|||
.main-body {flex: 2; border-left: 1px solid #e0e0e0;border-right: 1px solid #e0e0e0;margin: 0;overflow: hidden; |
|||
.empty-tips {text-align: center;width: 100%;font-size: 20px;color: #ccc;position: absolute;left: 0;top: 100px; |
|||
} |
|||
.main-form { background: #fff;border: 1px dashed #999;margin: 10px;height: calc(100vh - 165px);overflow-y: auto;overflow-x: hidden;position: relative} |
|||
} |
|||
// 中间按钮工具 |
|||
.main-tools {line-height: 26px;border-bottom: 2px solid #e4e7ed;margin-right: 10px;padding: 8px 0;display: flex;align-items: center;justify-content: flex-end; |
|||
button {color: $mainColor; |
|||
i {padding-right: 5px} |
|||
} |
|||
} |
|||
.add-form {padding: 0px;box-sizing: border-box; |
|||
// 格珊 |
|||
.form-row { |
|||
.form-col { |
|||
&.active-col {border: 3px solid $mainColor;position: relative; |
|||
> .drag-control {display: block;} |
|||
> .tooltip {display: block;} |
|||
} |
|||
} |
|||
} |
|||
.title {border-bottom: 1px solid #ddd;font-weight: 700;font-size: 14px;height: 30px;padding: 0 5px;margin-bottom: 22px} |
|||
.form-tabs { |
|||
.drag {min-height: 80px;} |
|||
} |
|||
.group-card { |
|||
.el-collapse {border: 0} |
|||
.el-collapse-item__header {font-weight: 700;border-bottom: 1px solid #ddd;margin-bottom: 20px} |
|||
.el-collapse-item__wrap {border: 0} |
|||
} |
|||
.group-inputSlot {display: none} |
|||
.form-table {margin-bottom: 22px; |
|||
.drag {border: 0;display: flex;overflow-x: auto;white-space: nowrap;padding: 0;flex-wrap: nowrap; |
|||
> div {min-width: 150px;width: auto} |
|||
} |
|||
.el-form-item {display: block} |
|||
} |
|||
.table-btn {padding-top: 10px} |
|||
.el-collapse-item__content {padding-bottom: 5px } |
|||
.form-table-add { |
|||
.el-form-item__label {display: none} |
|||
.el-form-item {margin: 0} |
|||
.el-table .cell {overflow: inherit; |
|||
.el-form-item__error {padding-top: 0} |
|||
} |
|||
} |
|||
.gray {color: #999} |
|||
> .drag {border: 0 !important;} |
|||
.group-flex { |
|||
.flex-group {display: flex;justify-content: space-between; |
|||
button {margin-left: 10px} |
|||
} |
|||
.flex-item {flex: 2;} |
|||
} |
|||
.group-txt {margin-bottom: 18px} |
|||
.div-layout { |
|||
/*&.inline { |
|||
.drag {display: inline-flex} |
|||
.group {width: auto;margin-right: 10px} |
|||
}*/ |
|||
&.right { |
|||
.drag {text-align: right;justify-content: flex-end} |
|||
} |
|||
&.center { |
|||
.drag {text-align: center;justify-content: center} |
|||
} |
|||
&.left { |
|||
|
|||
} |
|||
} |
|||
// 设置默认下input和select一样宽 |
|||
.el-select {width: 100%;} |
|||
.el-select { |
|||
.el-input__wrapper .el-input__suffix {position: absolute;right: 10px} |
|||
} |
|||
.group {width: 100%} |
|||
.drag {display: flex;flex-wrap: wrap} |
|||
&.form-row-2 { |
|||
.group {width: 50%} |
|||
} |
|||
&.form-row-3 { |
|||
.group {width: 33%} |
|||
} |
|||
&.form-row-4 { |
|||
.group {width: 25%} |
|||
} |
|||
} |
|||
.sidebar-tools {width: 300px;right: 0;top: 0;overflow-y: auto;box-sizing: border-box;padding-bottom: 10px;bottom: 0; height: calc(100vh - 60px); |
|||
.form { |
|||
.el-form-item {margin-bottom: 10px;} |
|||
} |
|||
.h3 {font-size: 14px;margin-bottom: 10px;display: flex;align-items: center; |
|||
h3 {font-size: 14px} |
|||
} |
|||
.el-tabs__nav-wrap {padding: 0 10px} |
|||
.el-tabs__content {padding: 0 10px} |
|||
.icon-del {cursor: pointer} |
|||
.option-radio { |
|||
> label {margin-right: 8px} |
|||
} |
|||
.event-btn { |
|||
button {margin: 0 12px 5px 0} |
|||
} |
|||
} |
|||
#editJson, #editJsonCopy {width: 100%;height: calc(100vh - 65px)} |
|||
#editJsonCopy {height: calc(100vh - 350px)} |
|||
.ace-dialog {background: #1e1e1e; |
|||
.el-drawer__body {padding: 0} |
|||
.el-drawer__header {margin: 0;color: #e9e9e9;font-size: 12px;padding: 3px 10px} |
|||
.dialog-footer {text-align: center;padding-top: 5px;} |
|||
} |
|||
.export-dialog { |
|||
.el-dialog__body {padding: 0 20px} |
|||
} |
|||
.design-form {min-height: calc(100vh - 170px); |
|||
> div {height: 100%} |
|||
> .drag {min-height: calc(100vh - 170px) !important;} |
|||
.ghost { |
|||
background: #F56C6C; |
|||
border: 2px solid #F56C6C; |
|||
outline-width: 0; |
|||
height: 3px; |
|||
box-sizing: border-box; |
|||
font-size: 0; |
|||
content: ''; |
|||
overflow: hidden; |
|||
padding: 0; |
|||
width: 100%; |
|||
} |
|||
.group {border: 1px dashed #ddd;margin: 2px 2px 10px 2px;padding: 5px;position: relative;min-height: 50px; |
|||
&.active {border: 3px solid $mainColor;position: relative; |
|||
> .drag-control {display: block} |
|||
> .tooltip {display: block;} |
|||
} |
|||
&:hover {border-color: $mainColor;background: #ecf5ff} |
|||
> div {margin-bottom: 0} |
|||
&:after{content: '';position: absolute;left: 0;top:0;right: 0;bottom: 0;opacity: 0;z-index: 1;display: block;} |
|||
&.group-grid,&.group-tabs,&.group-card,&.group-flex,&.group-div,&.group-table{ |
|||
&:after{display: none} |
|||
} |
|||
} |
|||
.group-inputSlot {border-color: #eebe77;display: inline-block; |
|||
&:hover, &.active {border-color: #eebe77 !important;} |
|||
} |
|||
.tooltip {display: none;position: absolute;font-size: 12px;top: 0;right: 0;z-index: 5} |
|||
.drag-control {display: none; |
|||
.item-control {position: absolute;right: 0;bottom: 0;z-index: 2; display: flex;align-items: center;height: 24px;background: $mainColor; |
|||
i {width: 24px;height: 24px;color: #fff;display: flex;align-items: center;justify-content: center;cursor: pointer} |
|||
} |
|||
.drag-move {position: absolute;left: 0;top: 0;z-index: 2;width: 24px;height: 24px;background: $mainColor;color: #fff;text-align: center;line-height: 24px;cursor: move} |
|||
} |
|||
.drag {height: 100%;border: 1px dashed #ddd;min-height: 40px;margin: 0 2px;padding: 5px;align-content: flex-start;} |
|||
&.form-row-2 { |
|||
.group {width: 48%} |
|||
} |
|||
&.form-row-3 { |
|||
.group {width: 32%} |
|||
} |
|||
&.form-row-4 { |
|||
.group {width: 23%} |
|||
} |
|||
} |
|||
/*表格设计*/ |
|||
.design-table { |
|||
.main-body {border-left: 0} |
|||
.header {position: relative; |
|||
.field {position: absolute;left: 10px;top: 7px;display: flex; |
|||
} |
|||
} |
|||
.components-list { |
|||
.content {padding: 0 12px; |
|||
> div {display: flex;flex-wrap: wrap;justify-content: flex-start; |
|||
label {margin: 0 10px 8px 0 !important;} |
|||
} |
|||
} |
|||
} |
|||
.main-table {padding: 10px 20px; |
|||
.add-form {min-height: auto;display: flex; |
|||
> .drag {height: auto;min-height: auto !important;} |
|||
} |
|||
.el-divider {margin: 8px 0 0} |
|||
.table-tip {color: #999;padding: 30px 0;line-height: 22px} |
|||
.search-box {position: relative; |
|||
&:after {content: '';width: 100%;height: 100%;position: absolute;left: 0;right: 0;bottom: 0;top: 0;z-index: 2;cursor: pointer;} |
|||
.group {width: auto;margin-right: 10px} |
|||
} |
|||
.control-btn {padding: 10px 0; |
|||
margin-bottom: 10px;display: flex;justify-content: space-between;align-items: center |
|||
} |
|||
.tip {border: 1px dotted #ddd;padding: 3px 5px;border-radius: 5px;color: #999;text-align: center;cursor: pointer} |
|||
.el-table__header { |
|||
th { |
|||
.cell {position: relative;} |
|||
.icon-close {display: none;font-size: 12px;margin-left: 12px;cursor: pointer;position: absolute;} |
|||
&:hover { |
|||
.icon-close {display: inline-block} |
|||
} |
|||
} |
|||
} |
|||
} |
|||
.table-tag { |
|||
.el-form-item__content {display: flex; |
|||
> div {flex: 2;margin-right: 5px;} |
|||
} |
|||
} |
|||
} |
|||
.table-field-list { |
|||
h3 {font-size: 14px} |
|||
.item {margin-bottom: 20px} |
|||
.list {display: flex;justify-content: flex-start;flex-wrap: wrap; |
|||
label {width: 25%;margin: 0} |
|||
} |
|||
} |
|||
.table-list-comm {display: flex;justify-content: space-between; |
|||
.tree-sidebar {width: 180px;border-right: 1px solid #ddd;padding-right: 10px;margin-right: 10px;flex-shrink: 0; |
|||
.el-input {margin-bottom: 10px} |
|||
} |
|||
.table-list {display: block;width: 100%} |
|||
.table-main {margin-bottom: 20px; |
|||
&.hide-vertical-scroll { // 固定了横向滚动条时,禁用表格固定头部的滚动 |
|||
.el-scrollbar__wrap {overflow: hidden;} |
|||
.is-vertical {display: none !important;} |
|||
} |
|||
.table-operate-btn {display: flex;align-items: center} |
|||
} |
|||
.table-search {position: relative; |
|||
form {display: flex;flex-wrap: wrap;padding-bottom: 0;border-bottom: 1px solid #dcdfe6;margin-bottom: 10px;} |
|||
.drag {display: flex;flex-wrap: wrap; |
|||
> div {margin-right: 10px} |
|||
} |
|||
.group {width: auto} |
|||
.search-icon {margin-left: 10px;height: 30px;cursor: pointer;position: absolute;right: 0;top: 0} |
|||
.autoHeight-enter-active, |
|||
.autoHeight-leave-active { |
|||
max-height: 200px; |
|||
transition: all .6s; |
|||
overflow: hidden; |
|||
} |
|||
.autoHeight-enter-from, |
|||
.autoHeight-leave-to { |
|||
max-height: 0; |
|||
} |
|||
} |
|||
.control-btn {margin-bottom: 10px;display: flex;justify-content: space-between;align-items: center} |
|||
} |
|||
/*图片文件上传*/ |
|||
.upload-style { |
|||
.limit { |
|||
.el-upload {display: none} |
|||
// 超出limit时不显示上传按钮 |
|||
} |
|||
.el-upload-list__item-preview {display: none !important;} |
|||
// 不显示点击放大按钮 |
|||
.el-upload--picture { |
|||
.icon-plus {font-size: 28px;display: flex;align-items: center;justify-content: center;color: #8c939d;width: 148px;height: 148px;border: 1px dashed #d9d9d9;border-radius: 6px;cursor: pointer;overflow: hidden;background: #fbfdff} |
|||
} |
|||
} |
|||
/*分页*/ |
|||
.table-page {padding-top: 10px;display: flex;justify-content: flex-end} |
|||
//.el-scrollbar__wrap{overflow: hidden!important;} |
|||
//.el-scrollbar__bar.is-vertical{display: none} |
|||
|
|||
@ -0,0 +1 @@ |
|||
@import "layout","form","flow","screen"; |
|||
@ -0,0 +1,120 @@ |
|||
html, body, div, span, ul, li, i, h3, p {margin: 0;padding: 0;} |
|||
ul, li {list-style: none} |
|||
body {font-size: 14px; font-family: "Microsoft YaHei UI"} |
|||
* {box-sizing: border-box} |
|||
a{color: #409eff;text-decoration: none} |
|||
::-webkit-scrollbar { |
|||
width: 8px; |
|||
height: 8px; |
|||
position: relative; |
|||
z-index: 100; |
|||
} |
|||
::-webkit-scrollbar-thumb { |
|||
background-color: #babdc2; |
|||
background-clip: padding-box; |
|||
min-height: 28px; |
|||
-webkit-border-radius: 5px; |
|||
-moz-border-radius: 5px; |
|||
border-radius: 5px; |
|||
} |
|||
::-webkit-scrollbar-corner { |
|||
background: #babdc2; |
|||
} |
|||
/*&::-webkit-scrollbar:$colorGray2; //滚动条整体部分 |
|||
::-webkit-scrollbar-button:''; //滚动条两端的按钮 |
|||
::-webkit-scrollbar-track:''; //外层轨道 |
|||
::-webkit-scrollbar-track-piece:''; //内层滚动槽 |
|||
::-webkit-scrollbar-thumb:$colorGray3; //滚动的滑块 |
|||
::-webkit-scrollbar-corner:''; //边角 |
|||
::-webkit-resizer //定义右下角拖动块的样式*/ |
|||
.common-layout { |
|||
.common-sidebar { |
|||
transition: all 0.3s; |
|||
box-shadow: 2px 0 6px rgb(0 21 41 / 35%); |
|||
color: #fff; |
|||
background-color: rgb(48 65 86); |
|||
height: 100vh; |
|||
overflow-x: hidden; |
|||
.logo { |
|||
display: flex; |
|||
align-items: center; |
|||
font-size: 14px; |
|||
height: 40px; |
|||
justify-content: center; |
|||
cursor: pointer; |
|||
font-weight: 700; |
|||
img { |
|||
width: 14px; |
|||
height: 14px; |
|||
} |
|||
} |
|||
.el-menu { |
|||
border-right: 0; |
|||
.icon{margin-right: 5px;font-size: 15px;opacity: .8} |
|||
} |
|||
.el-sub-menu .el-menu-item{min-width: 180px} |
|||
} |
|||
.common-header { |
|||
box-shadow: 0 1px 4px rgba(0, 21, 41, 0.08); |
|||
display: flex; |
|||
align-items: center; |
|||
justify-content: space-between; |
|||
.collapse-icon { |
|||
cursor: pointer; |
|||
font-size: 18px; |
|||
margin-right: 15px; |
|||
} |
|||
.breadcrumb {flex: 2;} |
|||
.comm-header-tool {margin: 0 10px;display: flex; |
|||
i {margin: 0 5px;cursor: pointer} |
|||
} |
|||
} |
|||
// 使用整个右侧滚动,没必要固定那个头部浪费空间 |
|||
.overflow-scroll{overflow-y: auto;display: block;height: 100vh} |
|||
.common-main { |
|||
background: #f0f2f5; |
|||
padding: 10px; |
|||
overflow: hidden; |
|||
> div:first-child {border-radius: 5px;padding: 10px 15px;background: #fff;overflow-x: auto; // 横向滚动条放在这里 |
|||
&.bg-none{background: none;} |
|||
} |
|||
} |
|||
.tag-tabs {display: flex;justify-content: space-between;align-items: center; |
|||
padding: 0 20px; |
|||
box-shadow: 0 1px 4px rgba(0, 21, 41, 0.08); |
|||
.el-tabs__content { |
|||
display: none; |
|||
} |
|||
.el-tabs__nav-wrap::after { |
|||
display: none; |
|||
} |
|||
.el-tabs__item { |
|||
padding: 0 10px !important; |
|||
} |
|||
.el-tabs__header { |
|||
margin: 0; |
|||
} |
|||
.tabs-label { |
|||
i {margin-left: 3px} |
|||
} |
|||
.clear-tag {cursor: pointer} |
|||
} |
|||
.quit-full {position: fixed;right: 10px;top: 10px;cursor: pointer;z-index: 101} |
|||
} |
|||
// 设置el-tooltip最大宽度 |
|||
.el-popper { |
|||
max-width: 400px; |
|||
} |
|||
.fade-transform-leave-active, |
|||
.fade-transform-enter-active { |
|||
transition: all .3s; |
|||
} |
|||
.fade-transform-enter-from{opacity: .3;transform: translateX(-30px)} |
|||
.fade-transform-enter-active { |
|||
} |
|||
.fade-transform-enter-to{transform: translateX(0);opacity: 1} |
|||
|
|||
.fade-transform-leave-active { |
|||
opacity: 0; |
|||
transform: translateX(30px); |
|||
} |
|||
@ -0,0 +1,111 @@ |
|||
.bg-black { /*background: rgb(48, 65, 86) !important;margin: -10px;border-radius: 0 !important;color: #fff*/ |
|||
} |
|||
|
|||
.container-screen {display: flex;justify-content: space-between;padding: 0 !important; |
|||
.tabs { |
|||
.el-tabs__nav {display: flex;justify-content: space-between;width: 100%} |
|||
.el-tabs__item {flex: 2;padding: 0;text-align: center} |
|||
} |
|||
.main-left {width: 180px;transition: all .3s; |
|||
.scroll {height: calc(100vh - 114px)} |
|||
.components-list {width: 100%; |
|||
ul { |
|||
li {padding: 0 5px; |
|||
i {font-size: 13px;margin-right: 3px} |
|||
} |
|||
} |
|||
} |
|||
.layer-list {font-size: 14px;overflow-y: auto; |
|||
ul {height: 100%} |
|||
li {line-height: 36px;padding: 0 8px;background: #f4f6fc;transition: all .3s;display: flex;justify-content: space-between;align-items: center;border: 1px solid transparent;border-bottom-color: #fff; |
|||
span {flex: 2;cursor: move;} |
|||
i {margin: 0 3px} |
|||
.icon {font-size: 20px;cursor: pointer} |
|||
&:hover {color: #66b1ff} |
|||
&.active { color: #fff;background: #66b1ff} |
|||
&.lock { color: #eebe77; |
|||
span {cursor: not-allowed} |
|||
} |
|||
&.display {opacity: .5} |
|||
&.isGroup{padding-left: 20px} |
|||
} |
|||
} |
|||
} |
|||
.main-box {flex: 2;border-left: 1px solid #e0e0e0;border-right: 1px solid #e0e0e0;height: calc(100vh - 66px);display: flex;flex-direction: column;color: #fff;width: 100%;position: relative;overflow: hidden; |
|||
.main-tools {line-height: 20px; |
|||
.control-tools {flex: 2;color: #999;margin-left: 20px; |
|||
i {margin-right: 5px;cursor: pointer} |
|||
.active {color: #409eff} |
|||
} |
|||
} |
|||
.design-box {background: rgb(48, 65, 86);flex: 2;padding: 0 0 0 20px;position: relative;overflow: auto;} |
|||
.draw-react {border: 1px solid #409eff;position: absolute;z-index: 999} |
|||
.design-canvas {height: 100%;position: absolute;cursor: pointer;top: 20px;left: 20px;transform-origin: left top; |
|||
.drag {height: 100%} |
|||
.resize-box { |
|||
position: absolute;left: 0;top: 0;width: 100%;height: 100%;border: 1px dotted #409eff;z-index: 5; |
|||
span {width: 8px;height: 8px;display: block;border-radius: 50%;background: #409eff;position: absolute;border: 1px solid rgba(255, 255, 255, .8); |
|||
} |
|||
.rs1 {left: -4px;top: -4px;cursor: se-resize;} |
|||
.rs2 {left: 50%;top: -4px;margin-left: -4px;cursor: s-resize} |
|||
.rs3 {right: -4px;top: -4px;cursor: sw-resize} |
|||
.rs4 {left: -4px;top: 50%;margin-top: -4px;cursor: e-resize} |
|||
.rs5 {right: -4px;top: 50%;margin-top: -4px;cursor: e-resize} |
|||
.rs6 {left: -4px;bottom: -4px;cursor: sw-resize} |
|||
.rs7 {left: 50%;bottom: -4px;margin-left: -4px;cursor: s-resize} |
|||
.rs8 {right: -4px;bottom: -4px;cursor: se-resize;} |
|||
&:hover {background: rgba(64, 158, 255, .5);cursor: move} |
|||
.position {font-size: 12px;position: absolute;left: 0;top: -20px} |
|||
} |
|||
&.preview {position: fixed;left: 0;top: 0;z-index: 50; |
|||
transform: scale(1) translate(0px, 0px) !important;transition: all .3s; |
|||
.resize-box {display: none} |
|||
} |
|||
} |
|||
.design-footer {background: rgb(48, 65, 86);display: flex;justify-content: space-between;padding: 0 5px;align-items: center; |
|||
.center {flex: 2;display: flex;justify-content: center} |
|||
.item {display: flex;align-items: center;margin: 0 10px; |
|||
.label {margin-right: 5px;display: block;white-space: nowrap} |
|||
} |
|||
.slider { |
|||
.label {margin-right: 10px} |
|||
.el-slider {width: 150px} |
|||
} |
|||
.icon {cursor: pointer} |
|||
//.offset{width: 120px;text-align: center;color: red} |
|||
} |
|||
.el-slider__marks-stop {background: red} |
|||
.no-date {color: #ccc;position: absolute;left: 50%;top: 30%;transform: translateX(-50%);font-size: 20px} |
|||
} |
|||
.main-right {width: 220px;transition: all .3s;position: relative; |
|||
form {padding: 0 10px} |
|||
.el-form-item {margin-bottom: 8px} |
|||
.color-picker { |
|||
.el-color-picker__trigger, .el-color-picker {width: 100%} |
|||
} |
|||
.upload-image { |
|||
.el-form-item__content {display: flex} |
|||
.el-input {flex: 2} |
|||
.el-upload-list {display: none} |
|||
} |
|||
&.lock { |
|||
&:after {position: absolute;left: 0;top: 0;bottom: 0;right: 0;content: '';background: rgba(255, 255, 255, .3)} |
|||
.lock {z-index: 2} |
|||
} |
|||
} |
|||
.close-preview {position: fixed;right: 10px;top: 10px;z-index: 51;color: #999;cursor: pointer} |
|||
.screen-right-menu {position: absolute;z-index: 100;background: #fff;width: 160px;transform: translateY(-50%); |
|||
li {padding: 5px 15px;cursor: pointer; |
|||
&:hover {background: #409eff;color: #fff} |
|||
&.display{cursor: not-allowed;opacity: .5; |
|||
&:hover{background: #fff;color: #333}} |
|||
} |
|||
} |
|||
} |
|||
|
|||
.design-canvas { |
|||
.group { |
|||
position: absolute; |
|||
} |
|||
.default-img, .default-bg {display: block;width: 100%;height: 100%} |
|||
} |
|||
@ -0,0 +1,322 @@ |
|||
const selectOption: any = [ |
|||
/*{ |
|||
label: '标签1', |
|||
value: 'value1' |
|||
}, |
|||
{ |
|||
label: '标签2', |
|||
value: 'value2' |
|||
}, |
|||
{ |
|||
label: '标签3', |
|||
value: 'value3' |
|||
}*/ |
|||
] |
|||
const config: { optionsType: number } = { |
|||
optionsType: 0 // 0固定 1数据源 2 接口字典
|
|||
} |
|||
export default [ |
|||
{ |
|||
title: '基础字段', |
|||
children: [ |
|||
{ |
|||
type: 'input', |
|||
label: '单行文本', |
|||
icon: 'input', |
|||
control: { |
|||
// 组件所有属性
|
|||
modelValue: '' |
|||
}, |
|||
config: {} // 其他配置信息
|
|||
}, |
|||
{ |
|||
type: 'textarea', |
|||
label: '多行文本', |
|||
icon: 'textarea', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'radio', |
|||
label: '单选框组', |
|||
icon: 'radio', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
options: selectOption, // 下拉选项数据集合
|
|||
config: config |
|||
}, |
|||
{ |
|||
type: 'checkbox', |
|||
label: '多选框组', |
|||
icon: 'checkbox', |
|||
control: { |
|||
modelValue: [] |
|||
}, |
|||
options: selectOption, |
|||
config: config |
|||
}, |
|||
{ |
|||
type: 'select', |
|||
label: '下拉选择框', |
|||
icon: 'select', |
|||
control: { |
|||
modelValue: '', |
|||
appendToBody: true |
|||
}, |
|||
options: selectOption, |
|||
config: config |
|||
}, |
|||
{ |
|||
type: 'datePicker', |
|||
label: '日期选择器', |
|||
icon: 'date', |
|||
control: { |
|||
modelValue: '', |
|||
type: 'date' |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'timePicker', |
|||
label: '时间选择器', |
|||
icon: 'time', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'colorPicker', |
|||
label: '取色器', |
|||
icon: 'color', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'switch', |
|||
label: '开关', |
|||
icon: 'switch', |
|||
control: { |
|||
modelValue: false |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'inputNumber', |
|||
label: '计数器', |
|||
icon: 'number', |
|||
control: { |
|||
modelValue: 0 |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'cascader', |
|||
label: '级联选择器', |
|||
icon: 'cascader', |
|||
control: { |
|||
modelValue: [] |
|||
}, |
|||
options: [], |
|||
config: config |
|||
}, |
|||
{ |
|||
type: 'rate', |
|||
label: '评分', |
|||
icon: 'rate', |
|||
control: { |
|||
modelValue: 0 |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'slider', |
|||
label: '滑块', |
|||
icon: 'slider', |
|||
control: { |
|||
modelValue: 0 |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'treeSelect', |
|||
label: '树形控件', |
|||
icon: 'tree2', |
|||
control: { |
|||
modelValue: '', |
|||
data: [], |
|||
renderAfterExpand: false |
|||
}, |
|||
config: { |
|||
optionsType: 0 |
|||
} |
|||
}, |
|||
{ |
|||
type: 'txt', |
|||
label: '文字', |
|||
icon: 'text', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'title', |
|||
label: '标题', |
|||
icon: 'title', |
|||
control: { |
|||
modelValue: '标题' |
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'button', |
|||
label: '按钮', |
|||
icon: 'button', |
|||
control: { |
|||
label: '保存' |
|||
}, |
|||
config: {} |
|||
} |
|||
] |
|||
}, |
|||
{ |
|||
title: '高级字段', |
|||
children: [ |
|||
{ |
|||
type: 'table', |
|||
label: '子表', |
|||
icon: 'table', |
|||
list: [], |
|||
tableData: [], // 子表表格列表数据集合
|
|||
control: { |
|||
border: true |
|||
}, |
|||
config: { |
|||
addBtnText: '添加一行' |
|||
} |
|||
}, |
|||
{ |
|||
type: 'component', |
|||
label: '自定义组件', |
|||
icon: 'component', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
config: {} |
|||
/*template: '', // 组件模板名称 |
|||
component: '' // 根据template注入的组件*/
|
|||
}, |
|||
{ |
|||
type: 'upload', |
|||
label: '图片/文件', |
|||
icon: 'image', |
|||
control: { |
|||
modelValue: '' // 也可以是[{name:'',url:''}]形式
|
|||
}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'tinymce', |
|||
label: 'tinymce富文本', |
|||
icon: 'tinymce', |
|||
control: { |
|||
modelValue: '' |
|||
}, |
|||
config: {} |
|||
} |
|||
] |
|||
}, |
|||
{ |
|||
title: '布局字段', |
|||
children: [ |
|||
{ |
|||
type: 'grid', |
|||
label: '格栅布局', |
|||
icon: 'grid', |
|||
columns: [ |
|||
// 格栅列数据
|
|||
{ |
|||
attr: { span: 12 }, |
|||
list: [] |
|||
}, |
|||
{ |
|||
attr: { span: 12 }, |
|||
list: [] |
|||
} |
|||
], |
|||
control: {}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'tabs', |
|||
label: '标签页', |
|||
icon: 'tabs', |
|||
columns: [ |
|||
{ |
|||
label: 'Tab1', |
|||
list: [] |
|||
} |
|||
], |
|||
control: {}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'card', |
|||
label: '卡片布局', |
|||
icon: 'card', |
|||
list: [], |
|||
control: {}, |
|||
config: {}, |
|||
item: {} // label标题相关
|
|||
}, |
|||
{ |
|||
type: 'flex', |
|||
label: '弹性布局', |
|||
icon: 'flex', |
|||
list: [], |
|||
tableData: [], // 值集合
|
|||
control: {}, |
|||
config: { |
|||
addBtnText: '添加一行' |
|||
} |
|||
}, |
|||
{ |
|||
type: 'divider', |
|||
label: '分割线', |
|||
icon: 'divider', |
|||
control: {}, |
|||
config: {} |
|||
}, |
|||
{ |
|||
type: 'div', |
|||
label: 'div容器', |
|||
icon: 'div', |
|||
control: {}, |
|||
config: {}, |
|||
list: [] |
|||
} |
|||
] |
|||
}, |
|||
{ |
|||
title: '扩展组件', |
|||
children: [ |
|||
{ |
|||
type: 'expand-user', |
|||
label: '选择用户', |
|||
icon: 'user', |
|||
control: { |
|||
// 组件所有属性
|
|||
modelValue: '' |
|||
}, |
|||
config: {} // 其他配置信息
|
|||
} |
|||
] |
|||
} |
|||
] |
|||
|
|||
@ -0,0 +1,166 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-12 10:58:26 |
|||
@ 备注: 快速选择表单字段 |
|||
--> |
|||
<script lang='ts' setup> |
|||
|
|||
|
|||
import { FormData,FormList } from '@/api/DesignForm/types' |
|||
import { stringToObj } from '@/utils/DesignForm/form' |
|||
import { jsonParseStringify } from '@/utils/DesignForm' |
|||
import controlListData from '@/components/DesignForm/assembly' |
|||
import { getRequest } from '@/api/DesignForm' |
|||
import Draggable from 'vuedraggable-es' |
|||
import UseTemplate from './template.vue' |
|||
const props = withDefaults( |
|||
defineProps<{ |
|||
formId?: number | string |
|||
}>(), |
|||
{} |
|||
) |
|||
// 默认搜索允许显示的字段 |
|||
const searchField = [ |
|||
'input', |
|||
'radio', |
|||
'checkbox', |
|||
'select', |
|||
'datePicker', |
|||
'timePicker', |
|||
'inputNumber', |
|||
'cascader', |
|||
'component', |
|||
'button' |
|||
] |
|||
const designType = inject('formDesignType') as string |
|||
const formDataList = ref<any[]>([]) |
|||
const isSearch = computed(() => { |
|||
return designType === 'search' |
|||
}) |
|||
const emits = defineEmits(["clickCheck","click"]); |
|||
// 筛选设计时左则勾选已有表单字段 |
|||
const selectChange = (val: any,obj: FormList) => { |
|||
if (val) { |
|||
// 勾选时追加 |
|||
const newObj = jsonParseStringify(obj) |
|||
delete newObj.rules |
|||
delete newObj.customRules |
|||
emits('clickCheck', newObj) |
|||
} |
|||
} |
|||
const controlList = computed(() => { |
|||
if (designType === 'search') { |
|||
// 只返回基础字段 |
|||
const temp: any = [] |
|||
controlListData.forEach((item: any) => { |
|||
if (item.children) { |
|||
const filter = item.children.filter((ch: any) => { |
|||
return searchField.includes(ch.type) |
|||
}) |
|||
if (filter && filter.length) { |
|||
temp.push({ title: item.title, children: filter }) |
|||
} |
|||
} |
|||
}) |
|||
return temp |
|||
} else { |
|||
return controlListData |
|||
} |
|||
}) |
|||
//使用模版 |
|||
const useTemplateEl = ref() |
|||
const useTemplateClick = () => { |
|||
useTemplateEl.value.open() |
|||
} |
|||
const useTemplateSelect = (data: FormData) => { |
|||
emits('click', data) |
|||
} |
|||
const clone = (origin: any) => { |
|||
return jsonParseStringify(origin) |
|||
} |
|||
//数据监听 |
|||
watch( |
|||
() => props.formId, |
|||
(val: any) => { |
|||
if (val && isSearch.value) { |
|||
getFormField(val) |
|||
} |
|||
} |
|||
) |
|||
// 加载当前列表所属的表单,从表单中提取可用于搜索的字段 |
|||
const getFormField = (formId: number) => { |
|||
getRequest('designById', { id: formId }).then((res: any) => { |
|||
const data = stringToObj(res.data.data) |
|||
if (data && data.list) { |
|||
forEachGetData(data.list) |
|||
} |
|||
}) |
|||
} |
|||
const forEachGetData = (data: FormList[]) => { |
|||
data.forEach((item: any) => { |
|||
if (item.type === 'grid' || item.type === 'tabs') { |
|||
item.columns.forEach((col: any) => { |
|||
forEachGetData(col.list) |
|||
}) |
|||
} else if (item.type === 'card') { |
|||
forEachGetData(item.list) |
|||
} else if (searchField.includes(item.type) && item.type !== 'button') { |
|||
formDataList.value.push(item) |
|||
} |
|||
}) |
|||
} |
|||
</script> |
|||
<template> |
|||
<div class="components-list"> |
|||
<div v-if="isSearch && formDataList?.length"> |
|||
<div class="title">快速选择表单字段</div> |
|||
<div class="content"> |
|||
<el-checkbox |
|||
v-for="item in formDataList" |
|||
:key="item.name" |
|||
@change="selectChange($event,item)" |
|||
> |
|||
{{ item.item?.label }} |
|||
</el-checkbox> |
|||
</div> |
|||
</div> |
|||
<!--组件主体--> |
|||
<div v-for="(list, index) in controlList" :key="index"> |
|||
<div class="title"> |
|||
{{ list.title }} |
|||
<div |
|||
class="template" |
|||
v-if="index === 0 && !isSearch" |
|||
@click="useTemplateClick" |
|||
> |
|||
使用模板 |
|||
</div> |
|||
</div> |
|||
<draggable |
|||
itemKey="key123" |
|||
tag="ul" |
|||
v-model="list.children" |
|||
:group="{ name: 'form', pull: 'clone', put: false }" |
|||
ghost-class="ghost" |
|||
:sort="false" |
|||
:clone="clone" |
|||
> |
|||
<template #item="{ element }"> |
|||
<li :class="[element.type]"> |
|||
<i :class="`icon-${element.icon}`"></i> |
|||
<span :title="element.label">{{ element.label }}</span> |
|||
</li> |
|||
</template> |
|||
</draggable> |
|||
</div> |
|||
<use-template |
|||
ref="useTemplateEl" |
|||
@click="useTemplateSelect" |
|||
v-if="!isSearch" |
|||
/> |
|||
|
|||
</div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,14 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-14 08:51:25 |
|||
@ 备注: 表单子表 |
|||
--> |
|||
<script lang='ts' setup> |
|||
|
|||
</script> |
|||
<template> |
|||
<div></div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,14 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-14 11:17:43 |
|||
@ 备注: |
|||
--> |
|||
<script lang='ts' setup> |
|||
|
|||
</script> |
|||
<template> |
|||
<div></div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,483 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-12 15:08:07 |
|||
@ 备注: 表单画布 |
|||
--> |
|||
<script lang='ts' setup> |
|||
import FormGroup from './formGroup.vue' |
|||
import { FormData,FormList } from '@/api/DesignForm/types' |
|||
import { getRequest } from '@/api/DesignForm' |
|||
import { useRoute, useRouter } from 'vue-router' |
|||
import { |
|||
constGetControlByName, |
|||
constSetFormOptions, |
|||
constFormBtnEvent, |
|||
constControlChange, |
|||
constFormProps, |
|||
appendOrRemoveStyle |
|||
} from '@/api/DesignForm/utils' |
|||
import formatResult from '@/utils/DesignForm/formatResult' |
|||
import formChangeValue from '@/utils/DesignForm/formChangeValue' |
|||
import { jsonParseStringify } from '@/utils/DesignForm' |
|||
|
|||
const props = withDefaults( |
|||
defineProps<{ |
|||
formData: FormData |
|||
type?: number // 1新增;2修改;3查看(表单模式) ;4查看; 5设计 |
|||
disabled?: boolean // 禁用表单提交 |
|||
requestUrl?: string // 编辑数据请求url |
|||
beforeRequest?: Function // 请求编辑数据前参数处理方法,可对请求参数处理 |
|||
afterResponse?: Function | string // 请求数据加载完成后数据处理方法,可对返回数据处理 |
|||
addUrl?: string // 表单数据新增提交保存url |
|||
editUrl?: string // 表单数据修改保存提交url |
|||
beforeSubmit?: Function | string // 表单提交前数据处理,可对提交数据处理,新增和保存都会触发 |
|||
afterSubmit?: Function // 表单提交后,默认提示提交结果,可return false阻止提示 |
|||
value?: { [key: string]: any } // 表单初始值,同setValue |
|||
options?: { [key: string]: any } // 表单组件选项,同setOptions |
|||
dict?: object // 固定匹配的字典 |
|||
isSearch?: boolean // 列表里作为筛选使用 |
|||
}>(), |
|||
{ |
|||
type: 1, // 1新增;2修改;3查看(表单模式) ;4查看; 5设计 |
|||
formData: () => { |
|||
return { |
|||
list: [], |
|||
form: {}, |
|||
config: {} |
|||
} |
|||
}, |
|||
dict: () => { |
|||
return {} |
|||
}, |
|||
isSearch: false |
|||
} |
|||
) |
|||
const emits = defineEmits<{ |
|||
(e: 'btnClick', type: string): void |
|||
(e: 'change', val: any): void // 表单组件值发生变化时 |
|||
}>() |
|||
|
|||
const route = useRoute() |
|||
const router = useRouter() |
|||
|
|||
const loading = ref(false) |
|||
|
|||
let timer = 0 |
|||
let eventName = '' |
|||
let getValueEvent = '' |
|||
// 注册window事件 |
|||
const setWindowEvent = (bool?: boolean) => { |
|||
if (props.formData.list.length > 0) { |
|||
const formName = props.formData.form?.name |
|||
if (!formName) { |
|||
// 导出.vue时,name可以没有 |
|||
return |
|||
} |
|||
eventName = `get${formName}ControlByName` |
|||
getValueEvent = `get${formName}ValueByName` |
|||
if (formName && (!window[eventName as any] || !bool)) { |
|||
// 根据name获取当前数据项 |
|||
// @ts-ignore |
|||
window[eventName] = (name: string) => { |
|||
return getNameForEach(props.formData.list, name) |
|||
} |
|||
// 根据name获取当前项的值 |
|||
// @ts-ignore |
|||
window[getValueEvent] = (name: string) => { |
|||
return model.value[name] |
|||
} |
|||
} |
|||
} |
|||
} |
|||
watch( |
|||
() => props.formData.config, |
|||
() => { |
|||
if (timer < 2) { |
|||
setWindowEvent() // 简单判断下,这里不是每次都更新 |
|||
} |
|||
timer++ |
|||
appendRemoveStyle(true) // 更新样式 |
|||
}, |
|||
{ deep: true } |
|||
) |
|||
setWindowEvent() |
|||
// 设置全局事件结束 |
|||
const resultDict = ref({}) |
|||
// 处理表单值开始 |
|||
const model = ref<any>({}) |
|||
// 获取表单初始model值 |
|||
const getInitModel = () => { |
|||
const obj = {} |
|||
forEachGetFormModel(props.formData.list, obj) |
|||
model.value = obj |
|||
} |
|||
watch( |
|||
() => props.formData.list, |
|||
() => { |
|||
// formData从接口获取时 |
|||
getInitModel() |
|||
} |
|||
) |
|||
// 从表单数据里提取表单所需的model |
|||
const forEachGetFormModel = (list: FormList[], obj: any) => { |
|||
list.forEach((item: any) => { |
|||
if (['table', 'flex'].includes(item.type)) { |
|||
obj[item.name] = jsonParseStringify(item.tableData) |
|||
} else if (['grid', 'tabs'].includes(item.type)) { |
|||
item.columns.forEach((col: any) => { |
|||
forEachGetFormModel(col.list, obj) |
|||
}) |
|||
} else if (['card', 'div'].includes(item.type)) { |
|||
forEachGetFormModel(item.list, obj) |
|||
} else { |
|||
const excludeType = ['title', 'divider', 'txt', 'button'] |
|||
if (excludeType.indexOf(item.type) === -1) { |
|||
obj[item.name] = jsonParseStringify(item.control.modelValue) |
|||
} |
|||
} |
|||
}) |
|||
} |
|||
|
|||
// 表单组件值改变事件 tProp为子表格相关 |
|||
provide(constControlChange, ({ key, value, data, tProp }: any) => { |
|||
if (key) { |
|||
if (!tProp) { |
|||
// 表格和弹性布局不是这里更新,只触change |
|||
model.value[key] = value |
|||
} |
|||
// 支持在线方式数据处理,如A组件值改变时,可自动修改B组件的值,可参考请假流程自动时长计算 |
|||
if (props.formData.events?.change) { |
|||
model.value = props.formData.events.change(key, model.value) |
|||
} |
|||
const onFormChange = props.formData.events?.change |
|||
if (onFormChange) { |
|||
if (typeof onFormChange === 'function') { |
|||
model.value = onFormChange(key, model.value) |
|||
} else { |
|||
model.value = formChangeValue(key, model.value, onFormChange) |
|||
} |
|||
} |
|||
// 当表格和弹性内的字段和外面字段冲突时,可通过tProps区分 |
|||
emits('change', { key, value, model: model.value, data, tProp }) |
|||
} |
|||
}) |
|||
const dictForm = computed(() => { |
|||
const storage = window.localStorage.getItem('akFormDict') |
|||
let storageDict = {} |
|||
if (storage) { |
|||
storageDict = JSON.parse(storage) |
|||
} |
|||
// 全局的、当前表单配置的以及接口返回的 |
|||
return Object.assign({}, storageDict, props.dict, resultDict.value) |
|||
}) |
|||
// 表单参数 |
|||
const formProps = computed(() => { |
|||
return { |
|||
model: model.value, |
|||
type: props.type, |
|||
hideField: props.formData.config?.hideField as [], |
|||
showColon: props.formData.form.showColon, |
|||
dict: dictForm.value |
|||
} |
|||
}) |
|||
provide(constFormProps, formProps) |
|||
|
|||
// 提供一个方法,用于根据name从formData.list里查找数据 |
|||
const getNameForEach = (data: any, name: string) => { |
|||
let temp = {} |
|||
for (const key in data) { |
|||
const dataKey = data[key] |
|||
if (dataKey.name === name) { |
|||
return dataKey |
|||
} |
|||
if (['grid', 'tabs'].includes(dataKey.type)) { |
|||
dataKey.columns.forEach((co: any) => { |
|||
temp = getNameForEach(co.list, name) |
|||
}) |
|||
} |
|||
if (['card', 'div'].includes(dataKey.type)) { |
|||
temp = getNameForEach(dataKey.list, name) |
|||
} |
|||
} |
|||
return temp |
|||
} |
|||
const getControlByName = (name: string) => { |
|||
return getNameForEach(props.formData.list, name) |
|||
} |
|||
provide(constGetControlByName, getControlByName) |
|||
// 表单检验方法 |
|||
const ruleForm = ref() |
|||
const validate = (callback: any) => { |
|||
ruleForm.value.validate((valid: boolean, fields: any) => { |
|||
let fieldValue = fields |
|||
if (valid) { |
|||
// 校验通过,返回当前表单的值 |
|||
fieldValue = getValue() |
|||
} |
|||
callback(valid, fieldValue) |
|||
}) |
|||
} |
|||
// 提供一个取值的方法 |
|||
const getValue = (filter?: boolean) => { |
|||
if (filter) { |
|||
const obj: any = {} |
|||
for (const key in model.value) { |
|||
// eslint-disable-next-line no-prototype-builtins |
|||
if (model.value.hasOwnProperty(key)) { |
|||
const val = (model.value as any)[key] |
|||
if (!/^\s*$/.test(val)) { |
|||
obj[key] = val |
|||
} |
|||
} |
|||
} |
|||
return obj |
|||
} else { |
|||
return model.value |
|||
} |
|||
} |
|||
// 对表单设置初始值 |
|||
const setValue = (obj: { [key: string]: any }, filter?: boolean) => { |
|||
// 分两种,false时将obj所有值合并到model,当obj有某些值不存于表单中,也会合并到model,当提交表单也会提交此值 |
|||
// true则过滤没用的值,即存在当前表单的才合并 |
|||
if (filter) { |
|||
for (const key in obj) { |
|||
if (model.value[key] !== undefined) { |
|||
model.value[key] = obj[key] |
|||
} |
|||
} |
|||
} else { |
|||
model.value = Object.assign({}, model.value, jsonParseStringify(obj)) // 防止列表直接使用set方法对弹窗表单编辑,当重置表单时当前行数据被清空 |
|||
} |
|||
} |
|||
// 对表单选择项快速设置 |
|||
const setFormOptions = ref({}) |
|||
provide(constSetFormOptions, setFormOptions) |
|||
const setOptions = (obj: { [key: string]: string[] }) => { |
|||
setFormOptions.value = obj |
|||
} |
|||
// 追加移除style样式 |
|||
const appendRemoveStyle = (type?: boolean) => { |
|||
try { |
|||
const { |
|||
config: { style } |
|||
} = props.formData |
|||
appendOrRemoveStyle('formStyle', style || '', type) |
|||
} catch (e) { |
|||
/* empty */ |
|||
} |
|||
} |
|||
|
|||
// 按钮组件事件 |
|||
provide(constFormBtnEvent, (obj: any) => { |
|||
emits('btnClick', obj.key) |
|||
if ([3, 4, 5].includes(props.type)) { |
|||
return ElMessage.error('当前模式不能提交表单') |
|||
} |
|||
switch (obj.key) { |
|||
case 'submit': |
|||
submit() // 提交 |
|||
break |
|||
case 'reset': |
|||
resetFields() // 重置 |
|||
break |
|||
case 'cancel': // 取消返回, |
|||
router.go(-1) //这个刷新后可能会失败 |
|||
break |
|||
} |
|||
}) |
|||
// 获取表单数据,编辑时,外部调用 |
|||
const getData = (params = {}) => { |
|||
const requestUrl = props.formData.config?.requestUrl || props.requestUrl |
|||
if (props.type === 5 || !requestUrl || props.isSearch) { |
|||
console.error('执行了获取数据方法,但配置有误!') |
|||
return |
|||
} |
|||
loading.value = true |
|||
const newParams: any = params |
|||
// 同时可使用props或是events里的事件,根据使用使用其中一种即可 |
|||
let newParams2 |
|||
const beforeRequest = props.formData.events?.beforeRequest |
|||
if (typeof beforeRequest === 'function') { |
|||
newParams2 = beforeRequest(newParams, route) |
|||
} |
|||
if (typeof props.beforeRequest === 'function') { |
|||
newParams2 = props.beforeRequest(newParams, route) |
|||
} |
|||
if (newParams2 === false) { |
|||
// 停止数据请求 |
|||
return |
|||
} |
|||
getRequest(requestUrl, newParams2 ?? newParams) |
|||
.then(res => { |
|||
// console.log(res) |
|||
loading.value = false |
|||
const result = res.data |
|||
if (result) { |
|||
let formatRes: any = result |
|||
// 比较适用通用表单,保存在服务端 |
|||
const afterResponse = props.formData.events?.afterResponse |
|||
if (typeof afterResponse === 'string' && afterResponse) { |
|||
formatRes = formatResult(result, afterResponse) |
|||
} else if (typeof afterResponse === 'function') { |
|||
formatRes = afterResponse(result) ?? result |
|||
} |
|||
// 比较适用于导出vue文件 |
|||
if (typeof props.afterResponse === 'string' && props.afterResponse) { |
|||
formatRes = formatResult(result, props.afterResponse) |
|||
} else if (typeof props.afterResponse === 'function') { |
|||
formatRes = props.afterResponse(result) ?? result |
|||
} |
|||
if (formatRes === false) { |
|||
return |
|||
} |
|||
setValue(formatRes.result || formatRes) |
|||
nextTick(() => { |
|||
// 将dict保存,可用于从接口中设置表单组件options。 |
|||
if (formatRes.dict && Object.keys(formatRes.dict).length) { |
|||
resultDict.value = formatRes.dict |
|||
} |
|||
}) |
|||
} |
|||
}) |
|||
.catch((res: any) => { |
|||
loading.value = false |
|||
return ElMessage.error(res.message) |
|||
}) |
|||
} |
|||
const submit = (params = {}) => { |
|||
const addUrl = props.formData.config?.addUrl || props.addUrl |
|||
const editUrl = props.formData.config?.editUrl || props.editUrl |
|||
const apiUrl = props.type === 1 ? addUrl : editUrl |
|||
if (props.isSearch || !apiUrl || loading.value) { |
|||
if (!props.isSearch && !apiUrl) { |
|||
console.error( |
|||
new Error('请在表单设计处配置接口事件url或选择数据源或设置props') |
|||
) |
|||
} |
|||
// 列表里作为筛选时,不提交表单 |
|||
return |
|||
} |
|||
validate((valid: boolean, fields: any) => { |
|||
if (valid) { |
|||
const formatParams = Object.assign({}, fields, params) |
|||
let submitParams |
|||
const beforeSubmit = props.formData.events?.beforeSubmit |
|||
if (beforeSubmit) { |
|||
if (typeof beforeSubmit === 'function') { |
|||
submitParams = beforeSubmit(formatParams, route) |
|||
} else { |
|||
submitParams = formatResult(formatParams, beforeSubmit) |
|||
} |
|||
} |
|||
if (props.beforeSubmit && typeof props.beforeSubmit === 'string') { |
|||
submitParams = formatResult(formatParams, props.beforeSubmit) |
|||
} else if (typeof props.beforeSubmit === 'function') { |
|||
submitParams = props.beforeSubmit(formatParams, route) |
|||
} |
|||
if (submitParams === false) { |
|||
return |
|||
} |
|||
loading.value = true |
|||
// 提交保存表单 |
|||
getRequest(apiUrl, submitParams ?? formatParams) |
|||
.then((res: any) => { |
|||
afterSubmit('success', res) |
|||
}) |
|||
.catch(res => { |
|||
afterSubmit('fail', res) |
|||
}) |
|||
} else { |
|||
// 没通过校验 |
|||
afterSubmit('validate', fields) |
|||
} |
|||
}) |
|||
} |
|||
// 不管成功失败,有事件时都需要执行回调 |
|||
const afterSubmit = (type: string, res: any) => { |
|||
const afterSubmit = props.formData.events?.afterSubmit |
|||
let notReturn |
|||
if (typeof afterSubmit === 'function') { |
|||
notReturn = afterSubmit(type, res) |
|||
} else if (typeof props.afterSubmit === 'function') { |
|||
notReturn = props.afterSubmit(type, res) |
|||
} |
|||
loading.value = false |
|||
// 不管结果,重置表单,防再次打开时保留上一次的值 |
|||
// resetFields() |
|||
if (notReturn === false) { |
|||
// 有返回false时则不提示 |
|||
return |
|||
} |
|||
if (type === 'success') { |
|||
ElMessage.success(res.message || '保存成功!') |
|||
} else if (type === 'fail') { |
|||
ElMessage.error(res.message || '保存失败!') |
|||
} |
|||
} |
|||
// 表单初始值 |
|||
watch( |
|||
() => props.value, |
|||
(v: any) => { |
|||
v && setValue(v) |
|||
}, |
|||
{ |
|||
immediate: true |
|||
} |
|||
) |
|||
// 表单options |
|||
watch( |
|||
() => props.options, |
|||
(v: any) => { |
|||
v && setOptions(v) |
|||
} |
|||
) |
|||
// ------------------------数据处理结束------------------------ |
|||
// 重置表单方法 |
|||
const resetFields = () => { |
|||
ruleForm.value.resetFields() |
|||
// setValue(Object.assign(model.value, obj || {})) // 这才能清空组件显示的值 |
|||
} |
|||
onMounted(() => { |
|||
getInitModel() |
|||
nextTick(() => { |
|||
appendRemoveStyle(true) |
|||
}) |
|||
}) |
|||
onUnmounted(() => { |
|||
if (eventName) { |
|||
// @ts-ignore |
|||
window[eventName] = '' |
|||
} |
|||
appendRemoveStyle() |
|||
}) |
|||
defineExpose({ |
|||
setOptions, |
|||
setValue, |
|||
getValue, |
|||
validate, |
|||
resetFields, |
|||
getData, |
|||
submit |
|||
}) |
|||
</script> |
|||
<template> |
|||
<div v-loading="loading"> |
|||
<el-form |
|||
v-bind="formData.form" |
|||
ref="ruleForm" |
|||
:model="model as any" |
|||
:disabled="disabled || type === 3" |
|||
class="add-form" |
|||
:class="{ |
|||
'design-form': type === 5, |
|||
'detail-form': type === 3 || type === 4 |
|||
}" |
|||
> |
|||
<FormGroup :data="formData.list" /> |
|||
<slot></slot> |
|||
</el-form> |
|||
</div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,436 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-14 08:46:51 |
|||
@ 备注: 表单组 |
|||
--> |
|||
<script lang='ts' setup> |
|||
import { computed, watch, inject, onUnmounted } from 'vue' |
|||
import Draggable from 'vuedraggable-es' |
|||
import FormItem from './formItem.vue' |
|||
import ChildTable from './childTable.vue' |
|||
import Tooltips from '@/components/DesignForm/tooltip.vue' |
|||
import FlexBox from './flexBox.vue' |
|||
import { useDesignFormStore } from '@/store/DesignForm/designForm' |
|||
import { FormList } from '@/api/DesignForm/types' |
|||
import { |
|||
constFormBtnEvent, |
|||
constFormProps |
|||
} from '@/api/DesignForm/utils' |
|||
import { Md5 } from 'ts-md5'; |
|||
import { jsonParseStringify } from '@/utils/DesignForm' |
|||
|
|||
const props = withDefaults( |
|||
defineProps<{ |
|||
data: FormList[] |
|||
}>(), |
|||
{ |
|||
data: () => { |
|||
return [] |
|||
} |
|||
} |
|||
) |
|||
const store = useDesignFormStore() as any |
|||
const formProps = inject(constFormProps, {}) as any |
|||
const type = computed(() => { |
|||
return formProps.value.type |
|||
}) |
|||
const state = reactive({ |
|||
clone: true, // 允许clone |
|||
gridAdd: false |
|||
}) |
|||
const dataList = ref<any>(props.data) |
|||
watch( |
|||
() => props.data, |
|||
(v: FormList[]) => { |
|||
dataList.value = v |
|||
} |
|||
) |
|||
const activeKey = computed(() => { |
|||
return store.activeKey |
|||
}) |
|||
|
|||
// 不能嵌套 |
|||
const notNested = (type: string) => { |
|||
const controlType = ['grid', 'table', 'tabs', 'div', 'flex', 'card'] |
|||
return controlType.includes(type) |
|||
} |
|||
// 删除或复制 |
|||
const click = (action: string, index: number, item?: any) => { |
|||
if (type.value !== 5) { |
|||
return // 非设计模式 |
|||
} |
|||
if (action === 'clone') { |
|||
const key = item.type + new Date().getTime().toString() |
|||
const newItem = jsonParseStringify(item) |
|||
dataList.value.splice(index, 0, Object.assign(newItem, { name: key })) |
|||
} else if (action === 'del') { |
|||
dataList.value.splice(index, 1) |
|||
// 清空右侧栏信息 |
|||
store.setActiveKey('') |
|||
store.setControlAttr({}) |
|||
} else if (action === 'gridAdd') { |
|||
item.columns.push({ |
|||
list: [], |
|||
attr: { span: 12 } |
|||
}) |
|||
} else if (action === 'delGridChild') { |
|||
item.splice(index, 1) |
|||
} |
|||
} |
|||
const draggableAdd = (evt: any) => { |
|||
if (type.value !== 5) { |
|||
return // 非设计模式 |
|||
} |
|||
const newIndex = evt.newIndex |
|||
const key = new Date().getTime().toString() |
|||
const obj: any = dataList.value[newIndex] |
|||
const isNested = evt.target && evt.target.getAttribute('data-type') // 不能嵌套 |
|||
if (isNested === 'not-nested' && notNested(obj.type)) { |
|||
dataList.value.splice(newIndex, 1) |
|||
return |
|||
} |
|||
if (!obj) { |
|||
return |
|||
} |
|||
const label = obj.label || obj.item.label |
|||
delete obj.label |
|||
delete obj.icon |
|||
let objectItem = {} |
|||
// 不需要添加item的项 |
|||
const notNeedItem = [ |
|||
'txt', |
|||
'title', |
|||
'button', |
|||
'table', |
|||
'grid', |
|||
'tabs', |
|||
'flex', |
|||
'div' |
|||
] |
|||
if (!notNeedItem.includes(obj.type)) { |
|||
objectItem = { |
|||
item: { |
|||
label: label |
|||
} |
|||
} |
|||
} |
|||
// 不需要name的组件 |
|||
let nameObj = {} |
|||
const notNeedName = [ |
|||
'txt', |
|||
'title', |
|||
'button', |
|||
'grid', |
|||
'tabs', |
|||
'divider', |
|||
'div', |
|||
'card' |
|||
] |
|||
if (!notNeedName.includes(obj.type) && !obj.name) { |
|||
nameObj = { |
|||
name: obj.type + key |
|||
} |
|||
} |
|||
Object.assign(obj, nameObj, objectItem) |
|||
groupClick(obj) |
|||
} |
|||
const getGroupName = (item: any) => { |
|||
if (item.name) { |
|||
return item.name |
|||
} else { |
|||
let md5Object:any = new Md5() |
|||
md5Object.appendAsciiStr(JSON.stringify(item)) |
|||
let md5Str = md5Object.end() |
|||
return md5Str |
|||
} |
|||
} |
|||
// 点击激活当前 |
|||
const groupClick = (item: any, ele?: string) => { |
|||
// 设计模式下才执行 |
|||
if (type.value !== 5) { |
|||
return |
|||
} |
|||
if (ele) { |
|||
item.type = ele |
|||
} |
|||
store.setActiveKey(getGroupName(item)) |
|||
store.setControlAttr(item) |
|||
// grid时显示添加列按钮 |
|||
state.gridAdd = item.type === 'grid' |
|||
state.clone = !notNested(item.type) |
|||
} |
|||
// 返回栅格宽度 |
|||
const getFormItemStyle = (ele: FormList) => { |
|||
if (ele.config?.span === 0) { |
|||
return { width: 'auto', margin: '0 5px' } |
|||
} |
|||
if (ele.config && ele.config.span) { |
|||
return { width: (ele.config.span / 24) * 100 + '%' } |
|||
} |
|||
} |
|||
// 根据关联条件显示隐藏当前项 |
|||
const linksShow = (el: FormList, index: number) => { |
|||
// 当前项设置了关联条件,当关联主体的值等于当前组件设定的值时 |
|||
if (!el.config) { |
|||
return true |
|||
} |
|||
const key = el.config.linkKey |
|||
const value = el.config.linkValue |
|||
const linkResult = el.config.linkResult |
|||
if (key && value && type.value !== 5) { |
|||
const Fn = new Function('$', `return (${value})`) |
|||
const pass = Fn(formProps.value.model) |
|||
if (linkResult === 'disabled') { |
|||
// 设置为disabled后返回显示状态 |
|||
dataList.value[index].control.disabled = pass |
|||
return true |
|||
} else { |
|||
return pass |
|||
} |
|||
} |
|||
return true |
|||
} |
|||
// 根据表单设置不显示指定字段 |
|||
const linksIf = (obj: FormList) => { |
|||
const { type } = formProps.value |
|||
const { config: { disabledAdd, disabledEdit, disabledDetail } = {} } = obj |
|||
if (type === 1) { |
|||
if (disabledAdd) { |
|||
// 编辑页 || 新增页 |
|||
return false // 不显示 |
|||
} |
|||
} else if (type === 2) { |
|||
// 编辑 |
|||
if (disabledEdit) { |
|||
return false |
|||
} |
|||
} else if (type === 4 || type === 3) { |
|||
// 查看 |
|||
if (disabledDetail) { |
|||
return false |
|||
} |
|||
} |
|||
// 如果当前字段的name值存在于表单数据的vIf中,则不显示 |
|||
const vIf: string | string[] = formProps.value.hideField |
|||
if (vIf?.length > 0 && obj.name) { |
|||
return vIf.indexOf(obj.name) === -1 // 存在时返回false隐藏 |
|||
} |
|||
return true |
|||
} |
|||
//按钮点击事件 |
|||
const injectBtnEvent = inject(constFormBtnEvent) |
|||
const clickBtn = (control: any) => { |
|||
// 0: '提交表单', |
|||
// 1: '重置表单', |
|||
// 2: '取消返回', |
|||
// 3: '无动作(自定义)' |
|||
if (type.value !== 5) { |
|||
// 非设计模式才触发事件 |
|||
injectBtnEvent && injectBtnEvent(control) |
|||
} |
|||
} |
|||
onUnmounted(() => { |
|||
// console.log('formGroup onUnmounted') |
|||
dataList.value = {} |
|||
store.setActiveKey('') |
|||
store.setControlAttr({}) |
|||
}) |
|||
</script> |
|||
<template> |
|||
<draggable |
|||
itemKey="id" |
|||
:list="dataList" |
|||
name="fade" |
|||
class="drag" |
|||
v-bind="{ |
|||
group: 'form', |
|||
ghostClass: 'ghost', |
|||
animation: 200, |
|||
handle: '.drag-move', |
|||
disabled: type !== 5 |
|||
}" |
|||
@add="draggableAdd" |
|||
> |
|||
<template #item="{ element, index }"> |
|||
<div |
|||
class="group" |
|||
:class="{ |
|||
['group-' + element.type]: true, |
|||
active: activeKey === getGroupName(element) |
|||
}" |
|||
:style="getFormItemStyle(element)" |
|||
@click.stop="groupClick(element)" |
|||
v-show="linksShow(element, index)" |
|||
v-if="linksIf(element)" |
|||
> |
|||
<template v-if="element.type === 'tabs'"> |
|||
<div class="form-tabs"> |
|||
<el-tabs |
|||
v-bind="element.control" |
|||
:class="[element.config?.className]" |
|||
> |
|||
<el-tab-pane |
|||
v-for="(item, tIndex) in element.columns" |
|||
:label="item.label" |
|||
:key="tIndex" |
|||
> |
|||
<form-group :data="item.list" data-type="not-nested" /> |
|||
</el-tab-pane> |
|||
</el-tabs> |
|||
</div> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'title'"> |
|||
<div |
|||
class="title" |
|||
:class="[element.config.className]" |
|||
v-bind="element.control" |
|||
> |
|||
<span v-html="element.control.modelValue"></span> |
|||
<Tooltips |
|||
:content="element.config.help" |
|||
v-if="element.config.help" |
|||
/> |
|||
</div> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'txt'"> |
|||
<div |
|||
v-bind="element.control" |
|||
:class="[element.config.className]" |
|||
v-html="element.control.modelValue" |
|||
></div> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'table'"> |
|||
<div class="form-table" v-if="type === 5"> |
|||
<form-group :data="element.list" data-type="not-nested" /> |
|||
</div> |
|||
<child-table v-else :data="element" /> |
|||
</template> |
|||
|
|||
|
|||
<template v-else-if="element.type === 'grid'"> |
|||
<el-row class="form-row" :class="[element.className]"> |
|||
<el-col |
|||
class="form-col" |
|||
:class="{ |
|||
'active-col': activeKey === getGroupName(col), |
|||
[col.className]: col.className |
|||
}" |
|||
v-bind="col.attr" |
|||
v-for="(col, i) in element.columns" |
|||
:key="i" |
|||
@click.stop="groupClick(col, 'gridChild')" |
|||
> |
|||
<form-group :data="col.list" data-type="not-nested" /> |
|||
<div class="drag-control" v-if="type === 5"> |
|||
<div class="item-control"> |
|||
<i |
|||
class="icon-del" |
|||
@click.stop=" |
|||
click('delGridChild', i as number, element.columns) |
|||
" |
|||
></i> |
|||
</div> |
|||
</div> |
|||
</el-col> |
|||
</el-row> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'card'"> |
|||
<el-collapse model-value="1"> |
|||
<el-collapse-item :title="element.item.label" name="1"> |
|||
<template #title v-if="element.help"> |
|||
{{ element.item.label }} |
|||
<Tooltips :content="element.help" /> |
|||
</template> |
|||
<form-group :data="element.list" data-type="not-nested" /> |
|||
</el-collapse-item> |
|||
</el-collapse> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'divider'"> |
|||
<el-divider v-bind="element.control" |
|||
>{{ element.item && element.item.label }} |
|||
</el-divider> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'div'"> |
|||
<div |
|||
class="div-layout" |
|||
v-bind="element.control" |
|||
:class="{ |
|||
[element.className]: element.className, |
|||
inline: element.config?.inline, |
|||
[element.config?.textAlign]: element.config?.textAlign |
|||
}" |
|||
> |
|||
<form-group :data="element.list" data-type="not-nested" /> |
|||
</div> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'flex'"> |
|||
<form-group |
|||
:data="element.list" |
|||
data-type="not-nested" |
|||
v-if="type === 5" |
|||
/> |
|||
<flex-box :data="element" v-else /> |
|||
<el-button |
|||
style="position: relative; top: -28px; left: 10px" |
|||
v-if="element.config.addBtnText && type === 5" |
|||
size="small" |
|||
>{{ element.config.addBtnText }}</el-button |
|||
> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'button'"> |
|||
<div |
|||
:class="[element.config?.className]" |
|||
:style="{ 'text-align': element.config?.textAlign }" |
|||
> |
|||
<el-button |
|||
v-bind="element.control" |
|||
@click="clickBtn(element.control)" |
|||
>{{ element.control?.label }}</el-button |
|||
> |
|||
</div> |
|||
</template> |
|||
|
|||
<template v-else-if="element.type === 'inputSlot' && type !== 5"> |
|||
<!-- 除设计外其他无需处理--> |
|||
</template> |
|||
|
|||
<form-item v-else :data="element" /> |
|||
|
|||
<template v-if="type === 5"> |
|||
<div class="drag-control"> |
|||
<div class="item-control"> |
|||
<i |
|||
class="icon-plus" |
|||
@click.stop="click('gridAdd', index, element)" |
|||
v-if="state.gridAdd" |
|||
title="添加列" |
|||
></i> |
|||
<i |
|||
class="icon-clone" |
|||
@click.stop="click('clone', index, element)" |
|||
v-if="state.clone" |
|||
title="克隆" |
|||
></i> |
|||
<i class="icon-del" @click.stop="click('del', index)"></i> |
|||
</div> |
|||
<div class="drag-move icon-move"></div> |
|||
</div> |
|||
<div class="tooltip">{{ element.name }}</div> |
|||
</template> |
|||
|
|||
|
|||
</div> |
|||
</template> |
|||
</draggable> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,14 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-14 08:50:36 |
|||
@ 备注: 表单项目 |
|||
--> |
|||
<script lang='ts' setup> |
|||
|
|||
</script> |
|||
<template> |
|||
<div></div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,68 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-12 14:41:18 |
|||
@ 备注: 主体画板顶部 |
|||
--> |
|||
<script lang='ts' setup> |
|||
import { computed } from 'vue' |
|||
|
|||
const props = withDefaults( |
|||
defineProps<{ |
|||
showKey?: string[] // showKey,hideKey设置其中一个即可,showKey优先 |
|||
hideKey?: string[] // 设置了showKey时,hideKey无效 |
|||
}>(), |
|||
{ |
|||
showKey: () => { |
|||
return [] |
|||
}, |
|||
hideKey: () => { |
|||
return [] |
|||
} |
|||
} |
|||
) |
|||
|
|||
const emits = defineEmits<{(e: 'click', value: string): void}>(); |
|||
|
|||
const btnClick = (type: string) => { |
|||
emits('click', type) |
|||
} |
|||
const btnList = computed(() => { |
|||
const list = [ |
|||
{ icon: 'del', label: '清空', key: 1 }, |
|||
{ icon: 'eye', label: '预览', key: 2 }, |
|||
{ icon: 'json', label: '生成脚本预览', key: 3 }, |
|||
{ icon: 'vue', label: '导出vue文件', key: 4 }, |
|||
{ icon: 'save', label: '保存', key: 5 } |
|||
] |
|||
if (props.showKey?.length) { |
|||
// 按照指定的key显示 |
|||
return list.filter((item: any) => { |
|||
return props.showKey.includes(item.key) |
|||
}) |
|||
} else if (props.hideKey?.length) { |
|||
// 按照指定的key隐藏 |
|||
return list.filter((item: any) => { |
|||
return !props.hideKey.includes(item.key) |
|||
}) |
|||
} else { |
|||
return list // 否则显示全部 |
|||
} |
|||
}) |
|||
</script> |
|||
<template> |
|||
<div class="main-tools"> |
|||
<slot></slot> |
|||
<el-button |
|||
link |
|||
type="primary" |
|||
@click="btnClick(item.icon)" |
|||
v-for="item in btnList" |
|||
:key="item.icon" |
|||
> |
|||
<i :class="['icon-' + item.icon]"></i>{{ item.label }} |
|||
</el-button> |
|||
</div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,49 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-14 08:27:12 |
|||
@ 备注: |
|||
--> |
|||
<script lang='ts' setup> |
|||
import { FormData,FileAttributeAll } from '@/api/DesignForm/types' |
|||
const emits = defineEmits<{ |
|||
(e: 'click', value: FormData): void |
|||
}>() |
|||
// const state = reactive({ |
|||
// visible: false, |
|||
// list: [] |
|||
// }) |
|||
const state = reactive<FileAttributeAll>({ |
|||
visible: false, |
|||
list: [] |
|||
}) |
|||
const open = () => { |
|||
state.visible = true |
|||
init() |
|||
} |
|||
const init = () => { |
|||
const template = import.meta.globEager('./template/*.ts') |
|||
console.log(template) |
|||
state.list = [] |
|||
Object.keys(template).forEach((key: string) => { |
|||
const file: any = template[key] |
|||
state.list.push({ |
|||
imgPath: file.imgPath, |
|||
title: file.title, |
|||
formData: file.formData |
|||
}) |
|||
}) |
|||
} |
|||
const selectClick = (item: any) => { |
|||
emits('click', item.formData) |
|||
state.visible = false |
|||
} |
|||
defineExpose({ |
|||
open |
|||
}) |
|||
</script> |
|||
<template> |
|||
<div></div> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,19 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-14 11:15:54 |
|||
@ 备注: |
|||
--> |
|||
<script lang='ts' setup> |
|||
withDefaults(defineProps<{ content: string }>(), {}) |
|||
</script> |
|||
<template> |
|||
<el-tooltip placement="top"> |
|||
<template #content> |
|||
<span v-html="content"></span> |
|||
</template> |
|||
<i class="icon-help"></i> |
|||
</el-tooltip> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,22 @@ |
|||
import { defineStore } from 'pinia' |
|||
|
|||
export const useDesignFormStore = defineStore('designForm', { |
|||
state: () => { |
|||
return { |
|||
controlAttr: {}, |
|||
activeKey: '' |
|||
//type: 1, //当前表单状态 1新增;2查看(表单模式) ;3查看; 4设计
|
|||
//isEdit: false, //编辑状态,type=1新增模式下有编辑状态,主要用于控制编辑模式下某些字段的禁用状态,即可新增但不能修改
|
|||
//model: {}, // 给form-group提供联动条件设置
|
|||
//hideField: [], // 设置了使用v-if隐藏的字段
|
|||
} |
|||
}, |
|||
actions: { |
|||
setControlAttr(data: any) { |
|||
this.controlAttr = data |
|||
}, |
|||
setActiveKey(key: string) { |
|||
this.activeKey = key |
|||
} |
|||
} |
|||
}) |
|||
@ -0,0 +1,42 @@ |
|||
import { defineStore } from 'pinia' |
|||
import { nextTick } from 'vue' |
|||
/*interface Breadcrumb { |
|||
label: string |
|||
to?: string |
|||
}*/ |
|||
interface TabsViews { |
|||
title: string |
|||
path: string |
|||
name: string // 路由名称name
|
|||
} |
|||
const getSession = window.sessionStorage.getItem('tagViews') |
|||
let tabs: any = [] |
|||
if (getSession) { |
|||
tabs = JSON.parse(getSession) |
|||
} |
|||
export const useLayoutStore = defineStore('layout', { |
|||
state: () => { |
|||
return { |
|||
breadcrumb: [], |
|||
tabs: tabs, |
|||
reloadFlag: true // 用于刷新路由
|
|||
} |
|||
}, |
|||
// 也可以定义为
|
|||
// state: () => ({ count: 0 })
|
|||
actions: { |
|||
changeBreadcrumb(data: any) { |
|||
this.breadcrumb = data |
|||
}, |
|||
setTabsViews(obj: TabsViews[]) { |
|||
this.tabs = obj |
|||
window.sessionStorage.setItem('tagViews', JSON.stringify(obj)) |
|||
}, |
|||
setReloadRouter() { |
|||
this.reloadFlag = false |
|||
nextTick(() => { |
|||
this.reloadFlag = true |
|||
}) |
|||
} |
|||
} |
|||
}) |
|||
@ -0,0 +1,87 @@ |
|||
// @ts-ignore
|
|||
import jsBeautify from 'js-beautify' |
|||
|
|||
export const EDITTYPE = 'javascript' // 弹出编辑器可输入类型 json/javascript
|
|||
|
|||
export function evil(fn: any) { |
|||
const Fn = Function // 一个变量指向Function,防止有些前端编译工具报错
|
|||
return new Fn('return ' + fn)() |
|||
} |
|||
|
|||
export function obj2string(o: any) { |
|||
let r: any = [] |
|||
if (o === null) { |
|||
// 这里有个问题 因typeOf null=object,下面判断会报错
|
|||
return null |
|||
} |
|||
if (typeof o === 'string') { |
|||
return ( |
|||
'"' + |
|||
o |
|||
.replace(/([\\'\\"\\])/g, '\\$1') |
|||
.replace(/(\n)/g, '\\n') |
|||
.replace(/(\r)/g, '\\r') |
|||
.replace(/(\t)/g, '\\t') + |
|||
'"' |
|||
) |
|||
} |
|||
if (typeof o === 'object') { |
|||
if (!o.sort) { |
|||
for (const i in o) { |
|||
if (o.hasOwnProperty(i)) { |
|||
let iii = i |
|||
if (i.indexOf('-') !== -1) { |
|||
iii = `"${i}"` |
|||
} |
|||
r.push(iii + ':' + obj2string(o[i])) |
|||
} |
|||
} |
|||
if ( |
|||
!!document.all && |
|||
!/^\n?function\s*toString\(\)\s*\{\n?\s*\[native code\]\n?\s*\}\n?\s*$/.test( |
|||
o.toString |
|||
) |
|||
) { |
|||
r.push('toString:' + o.toString.toString()) |
|||
} |
|||
r = '{' + r.join() + '}' |
|||
} else { |
|||
for (let i = 0; i < o.length; i++) { |
|||
r.push(obj2string(o[i])) |
|||
} |
|||
r = '[' + r.join() + ']' |
|||
} |
|||
return r |
|||
} |
|||
return o && o.toString() |
|||
} |
|||
|
|||
export function objToStringify(obj: any, isBeautify?: boolean) { |
|||
if (EDITTYPE === 'javascript') { |
|||
if (isBeautify) { |
|||
return jsBeautify('opt=' + obj2string(obj), { |
|||
indent_size: 2, |
|||
brace_style: 'expand' |
|||
}) |
|||
} else { |
|||
return obj2string(obj) |
|||
} |
|||
} else { |
|||
return isBeautify ? JSON.stringify(obj, null, 2) : JSON.stringify(obj) |
|||
} |
|||
} |
|||
|
|||
export function stringToObj(string: string) { |
|||
if (EDITTYPE === 'javascript') { |
|||
return evil(string) |
|||
} else { |
|||
return JSON.parse(string) |
|||
} |
|||
} |
|||
|
|||
export function string2json(string: string) { |
|||
return JSON.parse(string || '{}') |
|||
} |
|||
export function json2string(obj: any, isBeautify?: boolean) { |
|||
return isBeautify ? JSON.stringify(obj, null, 2) : JSON.stringify(obj) |
|||
} |
|||
@ -0,0 +1,16 @@ |
|||
// 用于在线模式表单值改变事件,此方法可实际如A组件改变时,自动对B组件设值或修改
|
|||
// 必须要return
|
|||
/* |
|||
@params name 当前改变的组件name值 |
|||
@params model当前表单的值 |
|||
@params key 当前设置的方法标识 |
|||
*/ |
|||
const formChangeValue = ( |
|||
name: string, |
|||
model: { [key: string]: any }, |
|||
key: string |
|||
) => { |
|||
console.log(key, model, name) |
|||
return model |
|||
} |
|||
export default formChangeValue |
|||
@ -0,0 +1,34 @@ |
|||
// 处理返回结果,当数据比较复杂时,使用在线编辑器如afterResponse方法不太适合,key即为afterResponse的字符串值
|
|||
// 必须要return
|
|||
const formatResult = (res: any, key: string) => { |
|||
// console.log('formatResult', res, key)
|
|||
switch (key) { |
|||
case 'transformDataToChild': |
|||
return transformDataToChild(res.list || res) |
|||
} |
|||
return res |
|||
} |
|||
// 根据id及parentId扁平数据转为children嵌套格式
|
|||
const transformDataToChild = (res: any) => { |
|||
const temp: any = [] |
|||
transformDataList(res, 0, temp) |
|||
return temp |
|||
} |
|||
const transformDataList = (data: any, parentId: number, temp: any) => { |
|||
data.forEach((item: any) => { |
|||
item.value = item.id // tree组件只能修改label取值,不能指定value,这里添加一个
|
|||
item.label = item.name |
|||
if (item.parentId === parentId) { |
|||
const childrenList = data.filter((ch: any) => { |
|||
return ch.parentId === item.id |
|||
}) |
|||
if (childrenList?.length) { |
|||
item.children = [] |
|||
transformDataList(data, item.id, item.children) |
|||
} |
|||
temp.push(item) |
|||
} |
|||
}) |
|||
} |
|||
// 部门侧栏树列表数据处理结束
|
|||
export default formatResult |
|||
@ -0,0 +1,18 @@ |
|||
// 用于在线模式表单值改变事件,此方法可实际如A组件改变时,自动对B组件设值或修改
|
|||
// 必须要return
|
|||
/* |
|||
@params key 当前指定的事件字符串标识 |
|||
@params result api接口返回结果 |
|||
@params options 静态或图片options数据,需按此格式返回 |
|||
@params global 全局的数据 |
|||
*/ |
|||
const formatScreen = ( |
|||
key: string, |
|||
result: any, |
|||
options?: any, |
|||
global?: any |
|||
) => { |
|||
console.log(key, result, options, global) |
|||
return options |
|||
} |
|||
export default formatScreen |
|||
@ -0,0 +1,95 @@ |
|||
export function debounce<T extends (...args: any[]) => void>(func: T, delay = 500, immediate?: boolean): T { |
|||
let timerId: any |
|||
|
|||
return function (this: any, ...args: any[]) { |
|||
if (timerId) { |
|||
clearTimeout(timerId) |
|||
} |
|||
if (immediate) { |
|||
const callNow = !timerId |
|||
timerId = setTimeout(() => { |
|||
timerId = null |
|||
}, delay) |
|||
if (callNow) { |
|||
func.apply(this, args) |
|||
} |
|||
} else { |
|||
timerId = setTimeout(() => { |
|||
func.apply(this, args) |
|||
}, delay) |
|||
} |
|||
} as T |
|||
} |
|||
// 时间格式化
|
|||
export const dateFormatting = (time: any, cFormat?: string) => { |
|||
const format = cFormat || '{y}-{m}-{d} {h}:{i}:{s}' |
|||
// 字符串数字形式的时间戳要转换下
|
|||
let newTime = time |
|||
if (/^\d+?$/.test(time)) { |
|||
newTime = parseInt(time) |
|||
} |
|||
const date = typeof time === 'object' ? time : new Date(newTime) |
|||
const formatObj: any = { |
|||
y: date.getFullYear(), |
|||
m: date.getMonth() + 1, |
|||
d: date.getDate(), |
|||
h: date.getHours(), |
|||
i: date.getMinutes(), |
|||
s: date.getSeconds(), |
|||
w: date.getDay() |
|||
} |
|||
return format.replace(/{(y|m|d|h|i|s|w)+}/g, (result, key) => { |
|||
let value = formatObj[key] |
|||
if (key === 'w') { |
|||
return ['日', '一', '二', '三', '四', '五', '六'][value] |
|||
} |
|||
if (result.length > 0 && value < 10) { |
|||
value = '0' + value |
|||
} |
|||
return value || 0 |
|||
}) |
|||
} |
|||
// 动态远程加载script脚本
|
|||
export function loadScript(src: string) { |
|||
return new Promise((resolve, reject) => { |
|||
const script = document.createElement('script') |
|||
script.type = 'text/javascript' |
|||
script.onload = resolve |
|||
script.onerror = reject |
|||
script.src = src |
|||
document.head.appendChild(script) |
|||
}) |
|||
} |
|||
// 随机数字符串
|
|||
export const randomString = (len: number) => { |
|||
len = len || 32 |
|||
const str = 'ABCDEFGHIJKMNOPQSTWXYZabcdefghijklmnopqrstwxyz1234567890' |
|||
let n = '' |
|||
for (let i = 0; i < len; i++) { |
|||
n += str.charAt(Math.floor(Math.random() * str.length)) |
|||
} |
|||
return n |
|||
} |
|||
|
|||
export const jsonParseStringify = (val: any) => { |
|||
if (typeof val === 'object') { |
|||
return JSON.parse(JSON.stringify(val)) |
|||
} else { |
|||
return val |
|||
} |
|||
} |
|||
/** |
|||
* 设置或获取local session storage |
|||
* @param key |
|||
* @param data 有值时set,否则get |
|||
* @param type local/session默认 |
|||
*/ |
|||
export const getSetStorage = (key: string, data?: string, type = 'session') => { |
|||
//console.log(key, data)
|
|||
const winType = type === 'session' ? 'sessionStorage' : 'localStorage' |
|||
if (data) { |
|||
window[winType].setItem(key, data) |
|||
} else { |
|||
return window[winType].getItem(key) |
|||
} |
|||
} |
|||
@ -0,0 +1,35 @@ |
|||
import axios from 'axios' |
|||
// 全局设置
|
|||
/* axios.defaults.baseURL = window.APP_CONFIG.BASE_URL |
|||
axios.defaults.headers.common['Authorization'] = getToken() */ |
|||
|
|||
const service = axios.create({ |
|||
baseURL: import.meta.env.VITE_APP_URL, // api的base_url
|
|||
//baseURL: 'http://localhost:3001/api', // api的base_url
|
|||
timeout: 3000, // request timeout
|
|||
headers: {} |
|||
}) |
|||
|
|||
service.interceptors.request.use( |
|||
(config) => { |
|||
return config |
|||
}, |
|||
(error) => { |
|||
Promise.reject(error) |
|||
} |
|||
) |
|||
|
|||
service.interceptors.response.use( |
|||
(response) => { |
|||
if (response.data.code === 1) { |
|||
return response.data |
|||
} else { |
|||
// 这里面需增加统一拦截
|
|||
return Promise.reject(response.data) |
|||
} |
|||
}, |
|||
(error) => { |
|||
return Promise.reject(error) |
|||
} |
|||
) |
|||
export default service |
|||
@ -0,0 +1,182 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-05 15:03:42 |
|||
@ 备注: 设置字段值 |
|||
--> |
|||
<script lang='ts' setup> |
|||
import { matrixTable,objectStruct } from '@/api/matrixapi/type' |
|||
import { getMatrixField } from '@/api/matrixapi'; |
|||
|
|||
const props = defineProps({ |
|||
userShow:{ |
|||
type:Boolean, |
|||
default:false |
|||
}, |
|||
matrixcont:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
}, |
|||
orglist:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
}, |
|||
orgtree:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
} |
|||
}); |
|||
const emits = defineEmits(["update:userShow"]); //父级元素 |
|||
const addFieldLoading = ref(false) |
|||
const tabelColumn = reactive<matrixTable[]>([]) |
|||
const tableList = reactive<any[]>([]) |
|||
const loading = ref(false) |
|||
|
|||
|
|||
const systemMenuTreeProps = { |
|||
children: "child", |
|||
label: "name", |
|||
value:"id" |
|||
} |
|||
const orgOptionsList = ref<any>(); //行政组织树 |
|||
/** |
|||
* 弹窗显示控制 |
|||
*/ |
|||
const field_is_Show = computed({ |
|||
get: () => props.userShow, |
|||
set: (val) => { |
|||
emits("update:userShow", val); |
|||
}, |
|||
}); |
|||
/** |
|||
* 关闭弹窗 |
|||
*/ |
|||
function handleCloseBox(){ |
|||
emits("update:userShow", false); |
|||
initData() |
|||
} |
|||
/** |
|||
* 初始化数据 |
|||
*/ |
|||
function initData(){ |
|||
orgOptionsList.value?.splice(orgOptionsList.value.length); |
|||
} |
|||
/** |
|||
* 监听数据 |
|||
*/ |
|||
watch(() => props.userShow,() => { |
|||
if(props.userShow){ |
|||
getMatrixField({id:props.matrixcont.id}) |
|||
.then(({data})=>{ |
|||
console.log("监听数据---->",data) |
|||
data.factor.forEach(item=>{ |
|||
tabelColumn.push({ |
|||
id:item.id, |
|||
label:item.name, |
|||
prop:item.pinyin, |
|||
types:item.types |
|||
}) |
|||
}) |
|||
data.outcome.forEach(item=>{ |
|||
tabelColumn.push({ |
|||
id:item.id, |
|||
label:item.name, |
|||
prop:item.pinyin, |
|||
types:item.types |
|||
}) |
|||
}) |
|||
}) |
|||
.finally(()=>{ |
|||
tableList.splice(0,tableList.length) |
|||
addTableHang(); |
|||
}) |
|||
} |
|||
}); |
|||
/** |
|||
* 添加行 |
|||
*/ |
|||
function addTableHang(){ |
|||
var jks:objectStruct = {} |
|||
tabelColumn.forEach(item=>{ |
|||
jks[item.prop]={ |
|||
id:item.id, |
|||
types:item.types, |
|||
number:1, |
|||
userlist:[309], |
|||
orgid:309, |
|||
namelist:["山东恒信高科能源有限公司"], |
|||
isedit:false |
|||
} |
|||
}) |
|||
tableList.push(jks) |
|||
// console.log("监听数据-1111->",props.orglist,props.orgtree) |
|||
} |
|||
/** |
|||
* 提交使用人 |
|||
*/ |
|||
function submitAddMatrixUser(){ |
|||
|
|||
} |
|||
</script> |
|||
<template> |
|||
<el-dialog v-model="field_is_Show" custom-class="dialog_box" title="矩阵数据维护" :before-close="handleCloseBox" width="80%"> |
|||
<el-table |
|||
v-loading="loading" |
|||
highlight-current-row |
|||
:data="tableList" |
|||
border |
|||
> |
|||
<el-table-column type="selection" width="55" align="center" /> |
|||
<el-table-column v-for="(item,index) in tabelColumn" :key="index" align="center"> |
|||
<!-- 自定义表头 --> |
|||
<template #header> |
|||
{{item.label}} |
|||
</template> |
|||
<!-- 自定义表项/单元格内容 --> |
|||
<template #default="scope"> |
|||
{{scope.row[item.prop].types}}---->{{scope.row[item.prop].isedit}} |
|||
<div v-if="!scope.row[item.prop].isedit" @dblclick="scope.row[item.prop].isedit = true"> |
|||
<el-text v-for="(itemtext,ti) in scope.row[item.prop].namelist" :key="ti" class="mx-1">{{ itemtext }}</el-text> |
|||
</div> |
|||
<div v-else> |
|||
<el-select v-if="scope.row[item.prop].types==3" v-model="scope.row[item.prop].orgid" clearable placeholder="请选择行政组织"> |
|||
<el-option |
|||
v-for="item in props.orglist.list" |
|||
:key="item.id" |
|||
:label="item.name" |
|||
:value="item.id" |
|||
/> |
|||
</el-select> |
|||
|
|||
<el-tree-select |
|||
v-if="scope.row[item.prop].types==2" |
|||
ref="orgTreePostRef" |
|||
v-model="scope.row[item.prop].orgid" |
|||
placeholder="选择归属行政组织" |
|||
:data="props.orgtree" |
|||
node-key="id" |
|||
check-strictly |
|||
:props="systemMenuTreeProps" |
|||
:render-after-expand="false" |
|||
class="orgTree" |
|||
/> |
|||
</div> |
|||
</template> |
|||
</el-table-column> |
|||
</el-table> |
|||
<template #footer> |
|||
<div class="dialog-footer"> |
|||
<el-button type="primary" :loading="addFieldLoading" @click="submitAddMatrixUser" >确 定</el-button> |
|||
<el-button @click="handleCloseBox">取 消</el-button> |
|||
</div> |
|||
</template> |
|||
</el-dialog> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
|
|||
</style> |
|||
@ -0,0 +1,613 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-07 09:11:55 |
|||
@ 备注: 选择行政组织 |
|||
--> |
|||
<script lang='ts' setup> |
|||
import { Search } from '@element-plus/icons-vue' //搜索图标 |
|||
import { ElDivider } from 'element-plus' |
|||
import { orgInfo,breadCrumbs,employeesCont,childDepartmentsCont } from '@/api/displayboardapi/types' |
|||
import { getOrgChiled,getBasisOrgChiled } from '@/api/displayboardapi/indexapi' |
|||
import { shuttleFramePickData } from '@/api/matrixapi/type' |
|||
|
|||
import { queryPeopleCont } from '@/api/hr/people/type' |
|||
import { searchUserCont } from '@/api/hr/people/index' |
|||
|
|||
|
|||
const spacer = h(ElDivider, { direction: 'vertical' }) |
|||
|
|||
const props = defineProps({ |
|||
orgBoxShow:{ |
|||
type:Boolean, |
|||
default:false |
|||
}, |
|||
ismultiselect:{ |
|||
type:Boolean, |
|||
default:false |
|||
}, |
|||
tabledata:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
} |
|||
}); |
|||
const emits = defineEmits(["update:orgBoxShow"]); //父级元素 |
|||
const dialogTitle = ref<string>() |
|||
const addOrgUserLoading = ref(false); |
|||
const searchKeywords = ref<string>() |
|||
const keywordsTitle = ref<string>() |
|||
const orgInfoList = ref<orgInfo[]>() |
|||
const orgBreadCrumbs = ref<breadCrumbs[]>() |
|||
const employees = ref<employeesCont[]>() |
|||
const childDepartments = ref<childDepartmentsCont[]>() |
|||
const isCheckBox = ref(false); //是否多选 |
|||
const pickDataList = ref<any>() |
|||
const duoxuanVal = reactive<shuttleFramePickData[]>([]) |
|||
// const danxuanVal = ref<shuttleFramePickData>() |
|||
const countPick=ref(0) |
|||
/** |
|||
* 弹窗显示控制 |
|||
*/ |
|||
const pickorg_is_Show = computed({ |
|||
get: () => props.orgBoxShow, |
|||
set: (val) => { |
|||
emits("update:orgBoxShow", val); |
|||
}, |
|||
}); |
|||
/** |
|||
* 关闭弹窗 |
|||
*/ |
|||
function closeBox(){ |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.isedit = false; |
|||
pickDataList.value =[] |
|||
orgInfoList.value = [] |
|||
duoxuanVal.slice(0,duoxuanVal.length) |
|||
let emptyOrg = new Array |
|||
duoxuanVal.length = 0 |
|||
searchKeywords.value="" |
|||
// //console.log("关闭弹窗",duoxuanVal) |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
// props.tabledata.namelist=["秦东","马飞"] |
|||
emits("update:orgBoxShow", false); |
|||
} |
|||
/** |
|||
* 获取行政组织 |
|||
*/ |
|||
function getOrgList(orgId:number){ |
|||
getOrgChiled({id:orgId}) |
|||
.then(({data})=>{ |
|||
// //console.log("获取行政组织",data) |
|||
orgInfoList.value = data.list |
|||
orgBreadCrumbs.value = data.tabval |
|||
}) |
|||
.finally(()=>{ |
|||
isPick() |
|||
}) |
|||
} |
|||
/** |
|||
* 判断是否选中 |
|||
*/ |
|||
function isPick(id?:number|string){ |
|||
// //console.log("orgInfoList->",orgInfoList) |
|||
// //console.log("pickDataList->",pickDataList) |
|||
if(pickDataList.value && pickDataList.value.length > 0){ |
|||
orgInfoList.value?.forEach(item=>{ |
|||
if(id == item.id){ |
|||
item.isActiveItem = false |
|||
} |
|||
pickDataList.value.forEach((items: any)=>{ |
|||
if(item.id == items.id){ |
|||
item.isActiveItem = true |
|||
} |
|||
}) |
|||
}) |
|||
employees.value?.forEach(item=>{ |
|||
if(id == item.id){ |
|||
item.ispick = false |
|||
} |
|||
pickDataList.value.forEach((items: any)=>{ |
|||
if(item.id == items.id){ |
|||
item.ispick = true |
|||
} |
|||
}) |
|||
}) |
|||
}else{ |
|||
orgInfoList.value?.forEach(item=>{ |
|||
if(id == item.id){ |
|||
item.isActiveItem = false |
|||
} |
|||
}) |
|||
employees.value?.forEach(items=>{ |
|||
if(id == items.id){ |
|||
items.ispick = false |
|||
} |
|||
}) |
|||
} |
|||
|
|||
} |
|||
/** |
|||
* 监听数据 |
|||
*/ |
|||
watch(() => props.orgBoxShow,() => { |
|||
duoxuanVal.slice(0,duoxuanVal.length) |
|||
if(props.orgBoxShow){ |
|||
isCheckBox.value = props.ismultiselect |
|||
pickDataList.value = props.tabledata.userlist |
|||
// //console.log("监听数据-----sd3------>",isCheckBox.value) |
|||
switch(props.tabledata.types){ |
|||
case 2: |
|||
dialogTitle.value = "选择行政组织" |
|||
keywordsTitle.value = "请输入关键字" |
|||
break; |
|||
case 3: |
|||
dialogTitle.value = "选择分部" |
|||
keywordsTitle.value = "请输入关键字" |
|||
break; |
|||
default: |
|||
dialogTitle.value = "选择成员" |
|||
keywordsTitle.value = "请输入姓名或工号" |
|||
} |
|||
if(props.tabledata.types > 1){ |
|||
getOrgList(313) |
|||
}else{ |
|||
getOrgAndPeople(313) |
|||
} |
|||
} |
|||
}) |
|||
/** |
|||
* 选定值 |
|||
*/ |
|||
function pickval(cont:any){ |
|||
|
|||
// //console.log("选定在----》",duoxuanVal) |
|||
if(isCheckBox.value){ |
|||
cont.isActiveItem = true |
|||
let xieRu = true; |
|||
duoxuanVal.forEach(item =>{ |
|||
if(item.id == cont.id){ |
|||
xieRu=false; |
|||
} |
|||
}) |
|||
if(xieRu){ |
|||
duoxuanVal.push({ |
|||
id:cont.id.toString(), |
|||
name:cont.name, |
|||
icon:"" |
|||
}) |
|||
pickDataList.value = duoxuanVal |
|||
} |
|||
}else{ |
|||
pickDataList.value =[ |
|||
{ |
|||
id:cont.id.toString(), |
|||
name:cont.name, |
|||
icon:"" |
|||
} |
|||
] |
|||
orgInfoList.value?.forEach(item=>{ |
|||
if(cont.id == item.id){ |
|||
item.isActiveItem = true |
|||
}else{ |
|||
item.isActiveItem = false |
|||
} |
|||
}) |
|||
} |
|||
countPick.value = pickDataList.value.length |
|||
|
|||
} |
|||
/** |
|||
* 选定人员 |
|||
*/ |
|||
function pickuserval(cont:any){ |
|||
if(cont.ispick){ |
|||
cont.ispick = false |
|||
let dxval = new Array |
|||
pickDataList.value.forEach((item: any)=>{ |
|||
if(item.id != cont.id){ |
|||
dxval.push(item); |
|||
} |
|||
}) |
|||
|
|||
duoxuanVal.splice(0,duoxuanVal.length) |
|||
dxval.forEach(items=>{ |
|||
duoxuanVal.push(items) |
|||
}) |
|||
pickDataList.value = dxval |
|||
}else{ |
|||
cont.ispick = true |
|||
let iconVal = cont.icon |
|||
if(iconVal == "" || iconVal == null) iconVal = cont.iconToBase64 |
|||
|
|||
duoxuanVal.push({ |
|||
id:cont.id.toString(), |
|||
name:cont.employeeName, |
|||
icon:iconVal |
|||
}) |
|||
pickDataList.value = duoxuanVal |
|||
//console.log("选定人员--1->",cont) |
|||
//console.log("选定人员--2->",pickDataList) |
|||
//console.log("选定人员--3->",duoxuanVal) |
|||
//console.log("选定人员--4->",iconVal) |
|||
} |
|||
countPick.value = pickDataList.value.length |
|||
} |
|||
/** |
|||
* 删除已经选择 |
|||
*/ |
|||
function delPickData(cont:any,index:number){ |
|||
pickDataList.value.splice(index, 1); |
|||
isPick(cont.id) |
|||
} |
|||
function delPickUserData(cont:any,index:number){ |
|||
pickDataList.value.splice(index, 1); |
|||
isPick(cont.id) |
|||
} |
|||
/** |
|||
* 提交数据 |
|||
*/ |
|||
function submitPickOrgOrUser(){ |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.isedit = false; |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
// props.tabledata.namelist=["秦东","马飞"] |
|||
if(pickDataList.value && pickDataList.value.length > 0){ |
|||
let nameAry = new Array |
|||
let userAry = new Array |
|||
pickDataList.value.forEach((item: any)=>{ |
|||
nameAry.push(item.name) |
|||
userAry.push({ |
|||
id:item.id.toString(), |
|||
name:item.name, |
|||
icon:item.icon |
|||
}) |
|||
}) |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.namelist = nameAry |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.userlist = userAry |
|||
}else{ |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.namelist = [] |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.userlist = [] |
|||
} |
|||
closeBox(); |
|||
} |
|||
/** |
|||
* 清空 |
|||
*/ |
|||
function emptyPickList(){ |
|||
pickDataList.value =[] |
|||
duoxuanVal.slice(0,duoxuanVal.length) |
|||
let emptyOrg = new Array |
|||
duoxuanVal.length = 0 |
|||
orgInfoList.value?.forEach(item=>{ |
|||
item.isActiveItem = false |
|||
}) |
|||
} |
|||
/** =========================================分割线。以下是行政组织人员相关内容=============================================================== */ |
|||
/** |
|||
* 清空已选择的人员 |
|||
*/ |
|||
function emptyPickuserList(){ |
|||
pickDataList.value =[] |
|||
duoxuanVal.slice(0,duoxuanVal.length) |
|||
let emptyOrg = new Array |
|||
duoxuanVal.length = 0 |
|||
employees.value?.forEach(item=>{ |
|||
item.ispick = false |
|||
}) |
|||
} |
|||
/** |
|||
* 获取行政组织和人员 |
|||
*/ |
|||
function getOrgAndPeople(orgid:number){ |
|||
getBasisOrgChiled({id:orgid.toString()}) |
|||
.then(({data})=>{ |
|||
//console.log("获取行政组织和人员",data) |
|||
orgBreadCrumbs.value = data.titleDepartments |
|||
employees.value = data.employees |
|||
childDepartments.value = data.childDepartments |
|||
}) |
|||
.finally(()=>{ |
|||
isPick() |
|||
}) |
|||
} |
|||
/** |
|||
* 搜索人员 |
|||
*/ |
|||
function searchPeople(){ |
|||
if(searchKeywords.value != ""){ |
|||
childDepartments.value = new Array |
|||
searchUserCont({page:1,pagesize:20,name:searchKeywords.value}) |
|||
.then(({data})=>{ |
|||
//console.log("搜索人员",data) |
|||
let usAry = new Array |
|||
data.list.forEach(item=>{ |
|||
let pickVal = false |
|||
if(pickDataList.value && pickDataList.value.length > 0){ |
|||
pickDataList.value.forEach((items: any)=>{ |
|||
if(item.id == items.id){ |
|||
pickVal = true |
|||
} |
|||
}) |
|||
} |
|||
|
|||
usAry.push({ |
|||
"id":item.id, //string //"95196156539179008", |
|||
"employeeName":item.employeeName, //string //"胡齐帅", |
|||
"isLeave":item.isLeave, //string //"0", |
|||
"open":item.open, //boolean //"false", |
|||
"icon":item.icon, //string //"", |
|||
"iconToBase64":item.iconToBase64, //string //"", |
|||
"wechat":item.wechat, //string //"", |
|||
"departmentid":item.departmentid, //number //102, |
|||
"departmentname":item.departmentname, //string //"企管部", |
|||
"postid":item.postid, //number //798, |
|||
"postname":item.postname, //string //"岗监员", |
|||
"tema":item.tema, //number //0, |
|||
"temaname":item.temaname, //string //"" |
|||
"ispick":pickVal, //boolean // |
|||
}) |
|||
}) |
|||
employees.value = usAry |
|||
}) |
|||
}else{ |
|||
getOrgAndPeople(313) |
|||
} |
|||
} |
|||
</script> |
|||
<template> |
|||
<el-dialog |
|||
v-model="pickorg_is_Show" |
|||
width="600px" |
|||
:title="dialogTitle" |
|||
append-to-body |
|||
draggable |
|||
:before-close="closeBox" |
|||
> |
|||
<el-row v-if="props.tabledata.types > 1" class="biankuang_all"> |
|||
<el-col :span="12" class="col_body "> |
|||
<div class="input_box"> |
|||
<el-input |
|||
v-model="searchKeywords" |
|||
:placeholder="keywordsTitle" |
|||
:prefix-icon="Search" |
|||
size="large" |
|||
class="inputDeep" |
|||
/> |
|||
</div> |
|||
<el-space :size="-2" :spacer="spacer" class="mianbaoxue"> |
|||
<el-text class="allSearch" @click="getOrgList(313)">所有</el-text> |
|||
<el-text v-for="(it,ind) in orgBreadCrumbs" :key="ind" @click="getOrgList(parseInt(it.id))">{{ it.departmentName }}</el-text> |
|||
</el-space> |
|||
<ul class="select-box"> |
|||
<li v-for="(item,index) in orgInfoList" :key="index" :class="{active: item.isActiveItem}"> |
|||
<div class="orgboxdiv"> |
|||
<el-tooltip |
|||
class="box-item" |
|||
effect="dark" |
|||
:content="item.name" |
|||
placement="left" |
|||
> |
|||
<el-text truncated @click="pickval(item)"> |
|||
<img src="@/assets/image/icon_file.png">{{ item.name }} |
|||
</el-text> |
|||
</el-tooltip> |
|||
<i v-if="item.ischild&&props.tabledata.types!=3" @click="getOrgList(item.id)">下级</i> |
|||
</div> |
|||
</li> |
|||
</ul> |
|||
</el-col> |
|||
<el-col :span="12" class="col_body biankuang_left"> |
|||
<div class="pick_head input_box"> |
|||
<span>已选择({{ countPick }})</span> |
|||
<span class="emptyPick" @click="emptyPickList">清空</span> |
|||
</div> |
|||
<ul class="pick_select_box"> |
|||
<li v-for="(item,index) in pickDataList" :key="index"> |
|||
<el-tooltip |
|||
class="box-item" |
|||
effect="dark" |
|||
:content="item.name" |
|||
placement="right" |
|||
> |
|||
<el-text truncated> |
|||
<img src="@/assets/image/icon_file.png">{{ item.name }} |
|||
</el-text> |
|||
</el-tooltip> |
|||
|
|||
<svg-icon icon-class="cc_k" class="right_5" @click="delPickData(item,index)" /> |
|||
</li> |
|||
</ul> |
|||
</el-col> |
|||
</el-row> |
|||
<!--选择人员--> |
|||
<el-row v-if="props.tabledata.types < 2" class="biankuang_all"> |
|||
<el-col :span="12" class="col_body "> |
|||
<div class="input_box"> |
|||
<el-input |
|||
v-model="searchKeywords" |
|||
:placeholder="keywordsTitle" |
|||
:prefix-icon="Search" |
|||
size="large" |
|||
class="inputDeep" |
|||
@input="searchPeople" |
|||
/> |
|||
</div> |
|||
<el-space :size="-2" :spacer="spacer" class="mianbaoxue"> |
|||
<el-text class="allSearch" @click="getOrgAndPeople(313)">所有</el-text> |
|||
<el-text v-for="(it,ind) in orgBreadCrumbs" :key="ind" @click="getOrgAndPeople(parseInt(it.id))">{{ it.departmentName }}</el-text> |
|||
</el-space> |
|||
<ul class="select-box"> |
|||
<li v-for="(item,index) in childDepartments" :key="index" :class="{active: item.isActiveItem}"> |
|||
<div class="orgboxdiv"> |
|||
<el-tooltip |
|||
class="box-item" |
|||
effect="dark" |
|||
:content="item.departmentName" |
|||
placement="left" |
|||
> |
|||
<el-text truncated> |
|||
<img src="@/assets/image/icon_file.png">{{ item.departmentName }} |
|||
</el-text> |
|||
</el-tooltip> |
|||
<i @click="getOrgAndPeople(parseInt(item.id))">下级</i> |
|||
</div> |
|||
</li> |
|||
<li v-for="(item,index) in employees" :key="index" :class="{active: item.ispick}"> |
|||
<div class="userboxdiv" @click="pickuserval(item)"> |
|||
<svg-icon v-if="item.ispick" icon-class="select3" /><svg-icon v-else icon-class="select1" /> |
|||
<img v-if="item.icon && item.icon != ''" :src="item.icon"> |
|||
<img v-else-if="item.icon == '' && item.iconToBase64 && item.iconToBase64 != ''" :src="item.iconToBase64"> |
|||
<img v-else src="@/assets/image/icon_people.png"> |
|||
{{ item.employeeName }} |
|||
</div> |
|||
</li> |
|||
</ul> |
|||
</el-col> |
|||
<el-col :span="12" class="col_body biankuang_left"> |
|||
<div class="pick_head input_box"> |
|||
<span>已选择({{ countPick }})</span> |
|||
<span class="emptyPick" @click="emptyPickuserList">清空</span> |
|||
</div> |
|||
<ul class="pick_select_box"> |
|||
<li v-for="(item,index) in pickDataList" :key="index"> |
|||
<el-tooltip |
|||
class="box-item" |
|||
effect="dark" |
|||
:content="item.name" |
|||
placement="right" |
|||
> |
|||
<el-text truncated> |
|||
<img v-if="item.icon && item.icon != ''" :src="item.icon"><img v-else src="@/assets/image/icon_people.png">{{ item.name }} |
|||
</el-text> |
|||
</el-tooltip> |
|||
|
|||
<svg-icon icon-class="cc_k" class="right_5" @click="delPickUserData(item,index)" /> |
|||
</li> |
|||
</ul> |
|||
</el-col> |
|||
</el-row> |
|||
|
|||
|
|||
<template #footer> |
|||
<div class="dialog-footer"> |
|||
<el-button type="primary" :loading="addOrgUserLoading" @click="submitPickOrgOrUser" >确 定</el-button> |
|||
<el-button @click="closeBox">取 消</el-button> |
|||
</div> |
|||
</template> |
|||
</el-dialog> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
.biankuang_all{ |
|||
border: 1px solid rgb(220, 223, 230); |
|||
} |
|||
.biankuang_left{ |
|||
border-left: 1px solid rgb(220, 223, 230); |
|||
} |
|||
.col_body{ |
|||
height: 502px; |
|||
padding:0 0px; |
|||
} |
|||
.input_box{ |
|||
border-bottom: 1px solid rgb(220, 223, 230); |
|||
} |
|||
.inputDeep { |
|||
:deep(.el-input__wrapper) { |
|||
box-shadow: 0 0 0 0px var(--el-input-border-color, var(--el-border-color)) inset; |
|||
cursor: default; |
|||
.el-input__inner { |
|||
cursor: default !important; |
|||
} |
|||
} |
|||
} |
|||
|
|||
.mianbaoxue{ |
|||
width: 100%; |
|||
overflow-x: auto; |
|||
white-space:nowrap; |
|||
padding: 5px 5px; |
|||
cursor: pointer; |
|||
} |
|||
.select-box { |
|||
height: 430px; |
|||
overflow-y: auto; |
|||
border-top: 1px solid rgb(220, 223, 230); |
|||
li{ |
|||
padding: 5px 0; |
|||
.userboxdiv{ |
|||
padding: 0 5px; |
|||
cursor: pointer; |
|||
} |
|||
.orgboxdiv{ |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
|
|||
} |
|||
i{ |
|||
min-width: 60px; |
|||
padding-left: 24px; |
|||
padding-right: 10px; |
|||
font-style: normal; |
|||
font-size: 12px; |
|||
cursor: pointer; |
|||
background: url(@/assets/image/next_level.png) no-repeat 10px center; |
|||
border-left: 1px solid rgb(238, 238, 238); |
|||
} |
|||
img { |
|||
width: 14px; |
|||
vertical-align: middle; |
|||
margin-right: 5px; |
|||
margin-left: 5px; |
|||
} |
|||
.el-text{ |
|||
cursor: pointer; |
|||
} |
|||
} |
|||
li.active{ |
|||
color: #3195f8; |
|||
i{ |
|||
background: url(@/assets/image/next_level_active.png) no-repeat 10px center; |
|||
} |
|||
.el-text{ |
|||
color: #3195f8; |
|||
} |
|||
} |
|||
} |
|||
.pick_head{ |
|||
width: 100%; |
|||
height: 41px; |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
padding: 0 5px; |
|||
} |
|||
.pick_select_box{ |
|||
height: 460px; |
|||
overflow-y: auto; |
|||
li{ |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
padding: 5px 0; |
|||
img { |
|||
width: 14px; |
|||
vertical-align: middle; |
|||
margin-right: 5px; |
|||
margin-left: 5px; |
|||
} |
|||
.right_5{ |
|||
min-width: 20px; |
|||
margin-right: 5px; |
|||
cursor: pointer; |
|||
} |
|||
} |
|||
} |
|||
.emptyPick{ |
|||
cursor: pointer; |
|||
} |
|||
</style> |
|||
@ -0,0 +1,256 @@ |
|||
<!-- |
|||
@ 作者: 秦东 |
|||
@ 时间: 2023-07-07 09:12:32 |
|||
@ 备注: 选择人员 |
|||
--> |
|||
<script lang='ts' setup> |
|||
import { Search,ArrowRight } from '@element-plus/icons-vue' //搜索图标 |
|||
import { ElDivider } from 'element-plus' |
|||
/** |
|||
* 引入图片 |
|||
*/ |
|||
|
|||
const spacer = h(ElDivider, { direction: 'vertical' }) |
|||
|
|||
const props = defineProps({ |
|||
orgBoxShow:{ |
|||
type:Boolean, |
|||
default:false |
|||
}, |
|||
tabledata:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
}, |
|||
orglist:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
}, |
|||
orgtree:{ |
|||
type:Object, |
|||
default(){ |
|||
return {} |
|||
} |
|||
} |
|||
}); |
|||
const emits = defineEmits(["update:orgBoxShow"]); //父级元素 |
|||
const dialogTitle = ref<string>() |
|||
const addOrgUserLoading = ref(false); |
|||
const searchKeywords = ref<string>() |
|||
const keywordsTitle = ref<string>() |
|||
/** |
|||
* 弹窗显示控制 |
|||
*/ |
|||
const pickorg_is_Show = computed({ |
|||
get: () => props.orgBoxShow, |
|||
set: (val) => { |
|||
emits("update:orgBoxShow", val); |
|||
}, |
|||
}); |
|||
/** |
|||
* 关闭弹窗 |
|||
*/ |
|||
function closeBox(){ |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.isedit = false; |
|||
// eslint-disable-next-line vue/no-mutating-props |
|||
props.tabledata.namelist=["秦东","马飞"] |
|||
emits("update:orgBoxShow", false); |
|||
} |
|||
/** |
|||
* 监听数据 |
|||
*/ |
|||
watch(() => props.orgBoxShow,() => { |
|||
if(props.orgBoxShow){ |
|||
console.log("监听数据-----sd3------>",props.tabledata) |
|||
switch(props.tabledata.types){ |
|||
case 2: |
|||
dialogTitle.value = "选择行政组织" |
|||
keywordsTitle.value = "请输入关键字" |
|||
break; |
|||
case 3: |
|||
dialogTitle.value = "选择分部" |
|||
keywordsTitle.value = "请输入关键字" |
|||
break; |
|||
default: |
|||
dialogTitle.value = "选择成员" |
|||
keywordsTitle.value = "请输入姓名或工号" |
|||
} |
|||
|
|||
} |
|||
}) |
|||
/** |
|||
* 提交数据 |
|||
*/ |
|||
function submitPickOrgOrUser(){} |
|||
</script> |
|||
<template> |
|||
<el-dialog |
|||
v-model="pickorg_is_Show" |
|||
width="600px" |
|||
:title="dialogTitle" |
|||
append-to-body |
|||
draggable |
|||
:before-close="closeBox" |
|||
> |
|||
<el-row class="biankuang_all"> |
|||
<el-col :span="12" class="col_body "> |
|||
<div class="input_box"> |
|||
<el-input |
|||
v-model="searchKeywords" |
|||
:placeholder="keywordsTitle" |
|||
:prefix-icon="Search" |
|||
size="large" |
|||
class="inputDeep" |
|||
/> |
|||
</div> |
|||
<el-space :size="-2" :spacer="spacer" class="mianbaoxue"> |
|||
<el-text class="allSearch">所有</el-text> |
|||
<el-text>恒信高科</el-text> |
|||
</el-space> |
|||
<ul class="select-box"> |
|||
<li class="active"> |
|||
<div class="orgboxdiv"> |
|||
<div> |
|||
<img src="@/assets/image/icon_file.png">恒信高科 |
|||
</div> |
|||
<i>下级</i> |
|||
</div> |
|||
</li> |
|||
<li class="active "> |
|||
<div class="userboxdiv"> |
|||
<svg-icon icon-class="select1" /><img src="@/assets/image/icon_people.png">秦东 |
|||
</div> |
|||
</li> |
|||
<li class="active "> |
|||
<div class="userboxdiv"> |
|||
<svg-icon icon-class="select3" /><img src="@/assets/image/icon_people.png">秦东 |
|||
</div> |
|||
</li> |
|||
|
|||
</ul> |
|||
</el-col> |
|||
<el-col :span="12" class="col_body biankuang_left"> |
|||
<div class="pick_head input_box"> |
|||
<span>已选择(1)</span> |
|||
<span>清空</span> |
|||
</div> |
|||
<ul class="pick_select_box"> |
|||
<li> |
|||
<div> |
|||
<img src="@/assets/image/icon_people.png">秦东 |
|||
</div> |
|||
<svg-icon icon-class="cc_k" class="right_5" /> |
|||
</li> |
|||
</ul> |
|||
</el-col> |
|||
</el-row> |
|||
<template #footer> |
|||
<div class="dialog-footer"> |
|||
<el-button type="primary" :loading="addOrgUserLoading" @click="submitPickOrgOrUser" >确 定</el-button> |
|||
<el-button @click="closeBox">取 消</el-button> |
|||
</div> |
|||
</template> |
|||
</el-dialog> |
|||
</template> |
|||
<style lang='scss' scoped> |
|||
.biankuang_all{ |
|||
border: 1px solid rgb(220, 223, 230); |
|||
} |
|||
.biankuang_left{ |
|||
border-left: 1px solid rgb(220, 223, 230); |
|||
} |
|||
.col_body{ |
|||
height: 502px; |
|||
padding:0 0px; |
|||
} |
|||
.input_box{ |
|||
border-bottom: 1px solid rgb(220, 223, 230); |
|||
} |
|||
.inputDeep { |
|||
:deep(.el-input__wrapper) { |
|||
box-shadow: 0 0 0 0px var(--el-input-border-color, var(--el-border-color)) inset; |
|||
cursor: default; |
|||
.el-input__inner { |
|||
cursor: default !important; |
|||
} |
|||
} |
|||
} |
|||
|
|||
.mianbaoxue{ |
|||
width: 100%; |
|||
overflow-x: auto; |
|||
white-space:nowrap; |
|||
padding: 5px 5px; |
|||
} |
|||
.select-box { |
|||
height: 430px; |
|||
overflow-y: auto; |
|||
border-top: 1px solid rgb(220, 223, 230); |
|||
li{ |
|||
padding: 5px 0; |
|||
.userboxdiv{ |
|||
padding: 0 5px; |
|||
cursor: pointer; |
|||
} |
|||
.orgboxdiv{ |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
|
|||
} |
|||
i{ |
|||
padding-left: 24px; |
|||
padding-right: 10px; |
|||
font-style: normal; |
|||
font-size: 12px; |
|||
cursor: pointer; |
|||
background: url(@/assets/image/next_level.png) no-repeat 10px center; |
|||
border-left: 1px solid rgb(238, 238, 238); |
|||
} |
|||
img { |
|||
width: 14px; |
|||
vertical-align: middle; |
|||
margin-right: 5px; |
|||
margin-left: 5px; |
|||
} |
|||
} |
|||
li.active{ |
|||
color: #3195f8; |
|||
i{ |
|||
background: url(@/assets/image/next_level_active.png) no-repeat 10px center; |
|||
} |
|||
} |
|||
} |
|||
.pick_head{ |
|||
width: 100%; |
|||
height: 41px; |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
padding: 0 5px; |
|||
} |
|||
.pick_select_box{ |
|||
height: 460px; |
|||
overflow-y: auto; |
|||
li{ |
|||
display: flex; |
|||
justify-content: space-between; |
|||
align-items: center; |
|||
padding: 5px 0; |
|||
img { |
|||
width: 14px; |
|||
vertical-align: middle; |
|||
margin-right: 5px; |
|||
margin-left: 5px; |
|||
} |
|||
.right_5{ |
|||
margin-right: 5px; |
|||
cursor: pointer; |
|||
} |
|||
} |
|||
} |
|||
</style> |
|||