@@ -2,7 +2,7 @@ import { store } from '/@/store/index.ts';
import { Session } from '/@/utils/storage';
import { NextLoading } from '/@/utils/loading';
import { setAddRoute, setFilterMenuAndCacheTagsViewRoutes } from '/@/router/index';
-import { demoRoutes, dynamicRoutes } from '/@/router/route';
+import { dynamicRoutes } from '/@/router/route';
import api from '/@/api/system';
@@ -43,7 +43,7 @@ export async function initBackEndControlRoutes() {
// 存储接口原始路由(未处理component),根据需求选择使用
store.dispatch('requestOldRoutes/setBackEndControlRoutes', JSON.parse(JSON.stringify(menuRoute)));
// 处理路由(component),替换 dynamicRoutes(/@/router/route)第一个顶级 children 的路由
- dynamicRoutes[0].children?.push(...await backEndComponent(menuRoute), ...demoRoutes);
+ dynamicRoutes[0].children?.push(...await backEndComponent(menuRoute));
// 添加动态路由
await setAddRoute();
// 设置递归过滤有权限的路由到 vuex routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
@@ -90,899 +90,6 @@ export const dynamicRoutes: Array<RouteRecordRaw> = [
},
];
-export const demoRoutes: Array<RouteRecordRaw> = [
- {
- path: '/demo',
- name: 'demo',
- component: () => import('/@/layout/routerView/parent.vue'),
- redirect: '/demo/system/menu',
- meta: {
- title: '案例演示',
- isLink: '',
- isHide: true,
- isKeepAlive: true,
- isAffix: false,
- isIframe: false,
- roles: ['admin'],
- icon: 'iconfont icon-diannao',
- },
- children: [
- path: '/demo/menu',
- name: 'menu',
- redirect: '/menu/menu1',
- title: 'message.router.menu',
- isHide: false,
- roles: ['admin', 'common'],
- icon: 'iconfont icon-caidan',
- path: '/demo/menu/menu1',
- name: 'menu1',
- redirect: '/menu/menu1/menu11',
- title: 'message.router.menu1',
- path: '/demo/menu/menu1/menu11',
- name: 'menu11',
- component: () => import('/@/views/demo/menu/menu1/menu11/index.vue'),
- title: 'message.router.menu11',
- path: '/demo/menu/menu1/menu12',
- name: 'menu12',
- redirect: '/menu/menu1/menu12/menu121',
- title: 'message.router.menu12',
- path: '/demo/menu/menu1/menu12/menu121',
- name: 'menu121',
- component: () => import('/@/views/demo/menu/menu1/menu12/menu121/index.vue'),
- title: 'message.router.menu121',
- path: '/demo/menu/menu1/menu12/menu122',
- name: 'menu122',
- component: () => import('/@/views/demo/menu/menu1/menu12/menu122/index.vue'),
- title: 'message.router.menu122',
- ],
- path: '/demo/menu/menu1/menu13',
- name: 'menu13',
- component: () => import('/@/views/demo/menu/menu1/menu13/index.vue'),
- title: 'message.router.menu13',
- path: '/demo/menu/menu2',
- name: 'menu2',
- component: () => import('/@/views/demo/menu/menu2/index.vue'),
- title: 'message.router.menu2',
- path: '/demo/fun',
- name: 'funIndex',
- redirect: '/fun/tagsView',
- title: 'message.router.funIndex',
- icon: 'iconfont icon-crew_feature',
- path: '/demo/fun/tagsView',
- name: 'funTagsView',
- component: () => import('/@/views/demo/fun/tagsView/index.vue'),
- title: 'message.router.funTagsView',
- icon: 'ele-Pointer',
- path: '/demo/fun/countup',
- name: 'funCountup',
- component: () => import('/@/views/demo/fun/countup/index.vue'),
- title: 'message.router.funCountup',
- icon: 'ele-Odometer',
- path: '/demo/fun/wangEditor',
- name: 'funWangEditor',
- component: () => import('/@/views/demo/fun/wangEditor/index.vue'),
- title: 'message.router.funWangEditor',
- icon: 'iconfont icon-fuwenbenkuang',
- path: '/demo/fun/cropper',
- name: 'funCropper',
- component: () => import('/@/views/demo/fun/cropper/index.vue'),
- title: 'message.router.funCropper',
- icon: 'iconfont icon-caijian',
- path: '/demo/fun/qrcode',
- name: 'funQrcode',
- component: () => import('/@/views/demo/fun/qrcode/index.vue'),
- title: 'message.router.funQrcode',
- icon: 'iconfont icon-ico',
- path: '/demo/fun/echartsMap',
- name: 'funEchartsMap',
- component: () => import('/@/views/demo/fun/echartsMap/index.vue'),
- title: 'message.router.funEchartsMap',
- icon: 'iconfont icon-ditu',
- path: '/demo/fun/printJs',
- name: 'funPrintJs',
- component: () => import('/@/views/demo/fun/printJs/index.vue'),
- title: 'message.router.funPrintJs',
- icon: 'ele-Printer',
- path: '/demo/fun/clipboard',
- name: 'funClipboard',
- component: () => import('/@/views/demo/fun/clipboard/index.vue'),
- title: 'message.router.funClipboard',
- icon: 'ele-DocumentCopy',
- path: '/demo/fun/gridLayout',
- name: 'funGridLayout',
- component: () => import('/@/views/demo/fun/gridLayout/index.vue'),
- title: 'message.router.funGridLayout',
- icon: 'iconfont icon-tuodong',
- path: '/demo/fun/splitpanes',
- name: 'funSplitpanes',
- component: () => import('/@/views/demo/fun/splitpanes/index.vue'),
- title: 'message.router.funSplitpanes',
- icon: 'iconfont icon--chaifenlie',
- path: '/demo/pages',
- name: 'pagesIndex',
- redirect: '/pages/filtering',
- title: 'message.router.pagesIndex',
- icon: 'iconfont icon-fuzhiyemian',
- path: '/demo/pages/filtering',
- name: 'pagesFiltering',
- component: () => import('/@/views/demo/pages/filtering/index.vue'),
- title: 'message.router.pagesFiltering',
- icon: 'ele-Sell',
- /**
- * 注意此处详情写法:
- * 1、嵌套进父级里时,面包屑显示为:首页/页面/过滤筛选组件/过滤筛选组件详情
- * 2、不嵌套进父级时,面包屑显示为:首页/页面/过滤筛选组件/过滤筛选组件详情
- * 3、想要父级不高亮,面包屑显示为:首页/页面/过滤筛选组件详情,设置路径为:/pages/filteringDetails
- */
- path: '/demo/pages/filtering/details',
- name: 'pagesFilteringDetails',
- component: () => import('/@/views/demo/pages/filtering/details.vue'),
- title: 'message.router.pagesFilteringDetails',
- isKeepAlive: false,
- icon: 'ele-Sunny',
- path: '/demo/pages/filtering/details1',
- name: 'pagesFilteringDetails1',
- component: () => import('/@/views/demo/pages/filtering/details1.vue'),
- title: 'message.router.pagesFilteringDetails1',
- path: '/demo/pages/iocnfont',
- name: 'pagesIocnfont',
- component: () => import('/@/views/demo/pages/iocnfont/index.vue'),
- title: 'message.router.pagesIocnfont',
- icon: 'ele-Present',
- path: '/demo/pages/element',
- name: 'pagesElement',
- component: () => import('/@/views/demo/pages/element/index.vue'),
- title: 'message.router.pagesElement',
- icon: 'ele-Eleme',
- path: '/demo/pages/awesome',
- name: 'pagesAwesome',
- component: () => import('/@/views/demo/pages/awesome/index.vue'),
- title: 'message.router.pagesAwesome',
- icon: 'ele-SetUp',
- path: '/demo/pages/formAdapt',
- name: 'pagesFormAdapt',
- component: () => import('/@/views/demo/pages/formAdapt/index.vue'),
- title: 'message.router.pagesFormAdapt',
- icon: 'iconfont icon-biaodan',
- path: '/demo/pages/tableRules',
- name: 'pagesTableRules',
- component: () => import('/@/views/demo/pages/tableRules/index.vue'),
- title: 'message.router.pagesTableRules',
- icon: 'iconfont icon-jiliandongxuanzeqi',
- path: '/demo/pages/formI18n',
- name: 'pagesFormI18n',
- component: () => import('/@/views/demo/pages/formI18n/index.vue'),
- title: 'message.router.pagesFormI18n',
- icon: 'iconfont icon-diqiu',
- path: '/demo/pages/formRules',
- name: 'pagesFormRules',
- component: () => import('/@/views/demo/pages/formRules/index.vue'),
- title: 'message.router.pagesFormRules',
- icon: 'iconfont icon-shuxing',
- path: '/demo/pages/listAdapt',
- name: 'pagesListAdapt',
- component: () => import('/@/views/demo/pages/listAdapt/index.vue'),
- title: 'message.router.pagesListAdapt',
- icon: 'iconfont icon-chazhaobiaodanliebiao',
- path: '/demo/pages/waterfall',
- name: 'pagesWaterfall',
- component: () => import('/@/views/demo/pages/waterfall/index.vue'),
- title: 'message.router.pagesWaterfall',
- icon: 'iconfont icon-zidingyibuju',
- path: '/demo/pages/steps',
- name: 'pagesSteps',
- component: () => import('/@/views/demo/pages/steps/index.vue'),
- title: 'message.router.pagesSteps',
- icon: 'iconfont icon-step',
- path: '/demo/pages/preview',
- name: 'pagesPreview',
- component: () => import('/@/views/demo/pages/preview/index.vue'),
- title: 'message.router.pagesPreview',
- icon: 'iconfont icon-15tupianyulan',
- path: '/demo/pages/waves',
- name: 'pagesWaves',
- component: () => import('/@/views/demo/pages/waves/index.vue'),
- title: 'message.router.pagesWaves',
- icon: 'iconfont icon-bolangneng',
- path: '/demo/pages/tree',
- name: 'pagesTree',
- component: () => import('/@/views/demo/pages/tree/index.vue'),
- title: 'message.router.pagesTree',
- icon: 'iconfont icon-shuxingtu',
- path: '/demo/pages/drag',
- name: 'pagesDrag',
- component: () => import('/@/views/demo/pages/drag/index.vue'),
- title: 'message.router.pagesDrag',
- path: '/demo/pages/lazyImg',
- name: 'pagesLazyImg',
- component: () => import('/@/views/demo/pages/lazyImg/index.vue'),
- title: 'message.router.pagesLazyImg',
- icon: 'ele-PictureFilled',
- path: '/demo/pages/dynamicForm',
- name: 'pagesDynamicForm',
- component: () => import('/@/views/demo/pages/dynamicForm/index.vue'),
- title: 'message.router.pagesDynamicForm',
- path: '/demo/pages/workflow',
- name: 'pagesWorkflow',
- component: () => import('/@/views/demo/pages/workflow/index.vue'),
- title: 'message.router.pagesWorkflow',
- icon: 'ele-Connection',
- path: '/demo/make',
- name: 'makeIndex',
- redirect: '/make/selector',
- title: 'message.router.makeIndex',
- icon: 'iconfont icon-siweidaotu',
- path: '/demo/make/selector',
- name: 'makeSelector',
- component: () => import('/@/views/demo/make/selector/index.vue'),
- title: 'message.router.makeSelector',
- icon: 'iconfont icon-xuanzeqi',
- path: '/demo/make/noticeBar',
- name: 'makeNoticeBar',
- component: () => import('/@/views/demo/make/noticeBar/index.vue'),
- title: 'message.router.makeNoticeBar',
- icon: 'ele-Bell',
- path: '/demo/make/svgDemo',
- name: 'makeSvgDemo',
- component: () => import('/@/views/demo/make/svgDemo/index.vue'),
- title: 'message.router.makeSvgDemo',
- icon: 'fa fa-thumbs-o-up',
- path: '/demo/params',
- name: 'paramsIndex',
- redirect: '/params/common',
- title: 'message.router.paramsIndex',
- icon: 'iconfont icon-zhongduancanshu',
- path: '/demo/params/common',
- name: 'paramsCommon',
- component: () => import('/@/views/demo/params/common/index.vue'),
- title: 'message.router.paramsCommon',
- icon: 'iconfont icon-putong',
- path: '/demo/params/common/details',
- name: 'paramsCommonDetails',
- component: () => import('/@/views/demo/params/common/details.vue'),
- title: 'message.router.paramsCommonDetails',
- icon: 'ele-Comment',
- path: '/demo/params/dynamic',
- name: 'paramsDynamic',
- component: () => import('/@/views/demo/params/dynamic/index.vue'),
- title: 'message.router.paramsDynamic',
- icon: 'iconfont icon-dongtai',
- path: '/demo/params/dynamic/details/:t/:id',
- name: 'paramsDynamicDetails',
- component: () => import('/@/views/demo/params/dynamic/details.vue'),
- title: 'message.router.paramsDynamicDetails',
- icon: 'ele-Lightning',
- path: '/demo/visualizing',
- name: 'visualizingIndex',
- redirect: '/visualizing/visualizingLinkDemo1',
- title: 'message.router.visualizingIndex',
- icon: 'ele-ChatLineRound',
- path: '/demo/visualizing/visualizingLinkDemo1',
- name: 'visualizingLinkDemo1',
- component: () => import('/@/layout/routerView/link.vue'),
- title: 'message.router.visualizingLinkDemo1',
- isLink: `./#/visualizingDemo1`,
- icon: 'iconfont icon-caozuo-wailian',
- path: '/demo/visualizing/visualizingLinkDemo2',
- name: 'visualizingLinkDemo2',
- title: 'message.router.visualizingLinkDemo2',
- isLink: `./#/visualizingDemo2`,
- path: '/demo/chart',
- name: 'chartIndex',
- component: () => import('/@/views/demo/chart/index.vue'),
- title: 'message.router.chartIndex',
- icon: 'iconfont icon-ico_shuju',
- path: '/demo/tools',
- name: 'tools',
- component: () => import('/@/views/demo/tools/index.vue'),
- title: 'message.router.tools',
- icon: 'iconfont icon-gongju',
- path: '/demo/link',
- name: 'layoutLinkView',
- title: 'message.router.layoutLinkView',
- isLink: 'https://element-plus.gitee.io/#/zh-CN/component/installation',
- path: '/demo/iframes',
- name: 'layoutIfameView',
- component: () => import('/@/layout/routerView/iframes.vue'),
- title: 'message.router.layoutIfameView',
- isLink: 'https://nodejs.org/zh-cn/',
- isIframe: true,
- icon: 'iconfont icon-neiqianshujuchucun',
-]
-
/**
* 定义静态路由
* @description 前端控制直接改 dynamicRoutes 中的路由,后端控制不需要修改,请求接口路由数据时,会覆盖 dynamicRoutes 第一个顶级 children 的内容(全屏,不包含 layout 中的路由出口)
@@ -996,33 +103,9 @@ export const staticRoutes: Array<RouteRecordRaw> = [
meta: {
title: '登录',
+ }
* 提示:写在这里的为全屏界面,不建议写在这里
* 请写在 `dynamicRoutes` 路由数组中
*/
- path: '/visualizingDemo1',
- name: 'visualizingDemo1',
- component: () => import('/@/views/demo/visualizing/demo1.vue'),
- path: '/visualizingDemo2',
- name: 'visualizingDemo2',
- component: () => import('/@/views/demo/visualizing/demo2.vue'),
- path: '/dbInit',
- name: 'dbInit',
- component: () => import('/@/views/dbInit/index.vue'),
- title: '系统初始化',
@@ -1,97 +0,0 @@
-<template>
- <el-card class="box-card">
- <template #header>
- <div class="card-header">
- <span>安装向导</span>
- </div>
- </template>
- <Step :number="1"></Step>
- <div class="content">
- <el-row :gutter="20">
- <el-col :span="6"><strong>环境检测</strong></el-col>
- <el-col :span="6"><strong>推荐配置</strong></el-col>
- <el-col :span="6"><strong>当前状态</strong></el-col>
- <el-col :span="6"><strong>最低要求</strong></el-col>
- </el-row>
- <el-col :span="6">操作系统</el-col>
- <el-col :span="6">类UNIX</el-col>
- <el-col :span="6">{{sysInfo.sysOsName}}</el-col>
- <el-col :span="6">不限制</el-col>
- <el-col :span="6">GO版本</el-col>
- <el-col :span="6"> >1.15 </el-col>
- <el-col :span="6">{{sysInfo.goVersion}}</el-col>
- <el-col :span="6">1.15</el-col>
- <div class="foot">
- <el-button @click="again">重新检测</el-button>
- <el-button type="primary" @click="next">下一步</el-button>
- </el-card>
-</template>
-<script lang="ts">
-import Step from './step.vue'
-import {getEnvInfo as fetchEnvInfo} from '/@/api/system/dbInit'
-import {ref, onMounted, defineComponent, inject} from "vue";
-export default defineComponent({
- name: "checkEnv",
- setup(){
- let sysInfo: any = ref({})
- const jump: any = inject('jump')
- const getEnvInfo = async () => {
- fetchEnvInfo().then((res:any) => {
- const {code, data} = res
- if (code === 0) {
- sysInfo.value = data
- }
- })
- onMounted(getEnvInfo)
- return {
- getEnvInfo,
- sysInfo,
- jump
- components:{
- Step
- methods:{
- next() {
- this.jump('Init')
- again() {
- this.getEnvInfo()
-})
-</script>
-<style scoped lang="scss">
-.box-card{
- margin:100px auto;
- width:500px;
- min-height:550px;
- .content {
- height:370px;
- margin-bottom:10px;
- padding:10px;
- overflow: auto;
- .el-row {
- margin-bottom: 10px;
- .foot {
- text-align:center;
-}
-</style>
@@ -1,44 +0,0 @@
- <h1>恭喜您, 安装完成!</h1>
- <p>..........................................</p>
- <div class="foot"><el-button type="primary" @click="accept">进入后台</el-button></div>
-<script>
-export default {
- name: "finish",
- accept() {
- this.$router.push({path:"/"})
- height:520px;
- border: 1px #E4E7ED solid;
- background-color:#F2F6FC;
@@ -1,194 +0,0 @@
- <Step :number="2"></Step>
- <el-form
- ref="ruleFormRef"
- status-icon
- label-width="120px"
- :model="ruleForm"
- :rules="rules"
- >
- <el-divider> 数据库配置 </el-divider>
- <el-form-item label="数据库地址" prop="dbHost">
- <el-input v-model="ruleForm.dbHost" placeholder="例如:127.0.0.1"/>
- </el-form-item>
- <el-form-item label="数据库端口" prop="dbPort">
- <el-input v-model.number="ruleForm.dbPort" placeholder="例如:3306"/>
- <el-form-item label="数据库用户名" prop="dbUser">
- <el-input v-model="ruleForm.dbUser" placeholder="请输入数据库用户名"/>
- <el-form-item label="数据库密码" prop="dbPass">
- <el-input v-model="ruleForm.dbPass" placeholder="请输入数据库密码"/>
- <el-form-item label="数据库名称" prop="dbName">
- <el-input v-model="ruleForm.dbName" placeholder="请输入数据库名称"/>
- <el-form-item label="数据库编码" prop="dbCharset">
- <el-autocomplete :fetch-suggestions="querySearch" v-model="ruleForm.dbCharset" placeholder="请输入数据库编码"/>
- <el-divider> Redis配置 </el-divider>
- <el-form-item label="Redis地址" prop="redisAddress">
- <el-input v-model="ruleForm.redisAddress" placeholder="例如:127.0.0.1"/>
- <el-form-item label="Redis端口" prop="redisPort">
- <el-input v-model.number="ruleForm.redisPort" placeholder="例如:6379"/>
- <el-form-item label="Redis索引" prop="redisDb">
- <el-input v-model.number="ruleForm.redisDb" placeholder="请输入Redis索引"/>
- <el-form-item label="Redis密码" prop="redisPass">
- <el-input v-model="ruleForm.redisPass" placeholder="请输入Redis密码"/>
- </el-form>
- <el-button type="primary" @click="jump('CheckEnv')">上一步</el-button>
- <el-button type="primary" @click="next(ruleFormRef)">创建数据</el-button>
-<script lang="ts" setup>
-import {createDb} from "/@/api/system/dbInit/index"
-import { reactive, ref, inject } from 'vue'
-const jump:any = inject('jump')
-import type { FormInstance, FormRules } from 'element-plus'
-import { ElMessage, ElLoading } from 'element-plus'
-const ruleFormRef = ref<FormInstance>()
-const ruleForm = reactive({
- dbHost: '127.0.0.1',
- dbPort: 3306,
- dbUser: 'root',
- dbPass:'root',
- dbName:'',
- dbCharset:'utf8mb4',
- redisAddress:'127.0.0.1',
- redisPort:6379,
- redisDb:1,
- redisPass:'',
-const rules = reactive<FormRules>({
- dbHost: [{ required: true, message: '请输入数据库地址', trigger: 'blur' }],
- dbPort: [{ required: true, message: '请输入数据库端口', trigger: 'blur' },{ type: 'number', message: '数据库端口必须是一个数值' }],
- dbUser: [{ required: true, message: '请输入数据库用户名', trigger: 'blur' }],
- dbName: [{ required: true, message: '请输入数据库名称', trigger: 'blur' }],
- dbCharset: [{ required: true, message: '请输入数据库编码' }],
- redisAddress: [{ required: true, message: '请输入Redis地址', trigger: 'blur' }],
- redisPort: [{ required: true, message: '请输入Redis端口', trigger: 'blur' },{ type: 'number', message: 'Redis端口必须是一个数值' }],
- redisDb: [{ required: true, message: '请输入Redis索引', trigger: 'blur' },{ type: 'number', message: 'Redis索引必须是一个数值' }],
-interface DbCharsetItem {
- value: string
-const dbCharsetList = ref<DbCharsetItem[]>([
- {value:'utf8mb4'},
- {value:'utf8'},
- {value:'gbk'},
- {value:'gb2312'}
-])
-const querySearch = (query: string, cb: any) => {
- const results = query ? dbCharsetList.value.filter((item:DbCharsetItem) => item.value.indexOf(query) >= 0) : dbCharsetList.value
- cb(results)
-const next = async (formEl: FormInstance | undefined) => {
- if (!formEl) return
- await formEl.validate(async (valid, fields) => {
- if (valid) {
- const loading = ElLoading.service({
- lock: true,
- text: 'Loading',
- background: 'rgba(0, 0, 0, 0.7)',
- let res:any = await createDb(ruleForm)
- loading.close()
- const {code, message, data} = res
- if (code === 0 && data === true) {
- ElMessage({
- showClose: true,
- message: '初始化成功',
- type: 'success',
- jump('Finish')
- } else {
- message: message || '初始化操作失败!',
- type: 'error',
-import { defineComponent} from "vue";
- name: "init",
- width:600px;
@@ -1,54 +0,0 @@
- <p>软件使用协议</p>
- <h3>免责声明:</h3>
- <div class="foot"><el-button type="primary" @click="accept">接 受</el-button></div>
-import {defineComponent} from "vue";
-import { inject } from 'vue'
- name: "protocol",
- const jump:any = inject('jump')
- this.jump('CheckEnv')
- <ul class="step">
- <li :class="['number', {'active': number >= 1}]"><span>1</span><em>检测环境</em></li>
- <li :class="['number', {'active': number >= 2}]"><span>2</span><em>创建数据</em></li>
- <li :class="['number', {'active': number >= 3}]"><span>3</span><em>完成安装</em></li>
- </ul>
- name: "step",
- props:{
- number:{
- type:Number,
- default: 1,
-.step {
- list-style-type:none;
- display: flex;
- justify-content:space-around;
- .active {
- span{
- background-color:#409EFF !important;
- em{
- font-weight: bold;
- .number {
- height:40px;
- line-height:40px;
- span {
- margin-right:10px;
- display:inline-block;
- width:20px;
- height:20px;
- line-height: 20px;
- text-align: center;
- background-color:#E4E7ED;
- color:#fff;
- border-radius: 50%;
@@ -1,45 +0,0 @@
- <div class="db-init">
- <component :is="currentComponent" />
-import Protocol from "./component/protocol.vue";
-import CheckEnv from "./component/checkEnv.vue";
-import Init from "./component/init.vue"
-import Finish from "./component/finish.vue"
- name: "dbInit",
- Protocol,
- CheckEnv,
- Init,
- Finish
- provide() {
- jump: (v:string) => {
- this.currentComponent = v
- setup() {
- data() {
- currentComponent: 'Protocol'
- .db-init {
@@ -1,434 +0,0 @@
-.chart-scrollbar {
- .chart-warp {
- flex-direction: column;
- height: 100%;
- .chart-warp-bottom {
- flex: 1;
- overflow: hidden;
- .big-data-down-left,
- .big-data-down-right {
- width: 30%;
- .flex-warp-item {
- padding: 0 7.5px 15px 15px;
- width: 100%;
- height: 33.33%;
- .flex-warp-item-box {
- background: var(--el-color-white);
- border: 1px solid var(--el-border-color-lighter);
- border-radius: 4px;
- padding: 15px;
- transition: all ease 0.3s;
- &:hover {
- box-shadow: 0 2px 12px 0 rgb(0 0 0 / 10%);
- .flex-title {
- margin-bottom: 15px;
- justify-content: space-between;
- .flex-title-small {
- font-size: 12px;
- .flex-content {
- .flex-content-overflow {
- .big-data-down-left {
- color: var(--el-text-color-primary);
- .sky {
- align-items: center;
- .sky-left {
- font-size: 30px;
- .sky-center {
- padding: 0 10px;
- font {
- margin-right: 15px;
- .span {
- background: #22bc76;
- border-radius: 2px;
- padding: 0 5px;
- color: var(--el-color-white);
- .sky-right {
- font-size: 20px;
- .sky-dd {
- .sky-dl {
- height: 28px;
- div {
- i {
- font-size: 14px;
- .tip {
- white-space: nowrap;
- text-overflow: ellipsis;
- .sky-dl-first {
- color: var(--el-color-primary);
- .d-states {
- .d-states-item {
- height: 33px;
- width: 33px;
- line-height: 33px;
- border-radius: 100%;
- flex-shrink: 1;
- justify-content: center;
- .i-bg1 {
- .i-bg2 {
- background: #e2356d;
- .i-bg3 {
- background: #43bbef;
- .d-states-flex {
- padding: 0 10px 0;
- .d-states-item-label {
- .d-states-item-value {
- margin-top: 3px;
- .d-btn {
- margin-top: 5px;
- .d-btn-item {
- border: 1px solid var(--el-color-primary);
- border-radius: 35px;
- padding: 5px;
- margin-top: 15px;
- cursor: pointer;
- .d-btn-item-left {
- width: 25px;
- height: 25px;
- line-height: 25px;
- .d-btn-item-center {
- .d-btn-item-eight {
- text-align: right;
- padding-right: 10px;
- .big-data-down-center {
- width: 40%;
- .big-data-down-center-one {
- height: 66.67%;
- padding: 0 7.5px 15px;
- .big-data-down-center-one-content {
- .big-data-down-center-two {
- height: calc(100% - 30px);
- .flex-content-left {
- flex-wrap: wrap;
- width: 120px;
- .monitor-item {
- width: 50%;
- .monitor-wave {
- width: 40px;
- height: 40px;
- position: relative;
- background-color: var(--el-color-primary);
- &::before,
- &::after {
- content: '';
- position: absolute;
- left: 50%;
- background: #f4f4f4;
- animation: roateOne 10s linear infinite;
- transform: translateX(-50%);
- z-index: 1;
- &::before {
- bottom: 10px;
- border-radius: 60%;
- bottom: 8px;
- opacity: 0.7;
- border-radius: 37%;
- .monitor-z-index {
- z-index: 2;
- @keyframes roateOne {
- 0% {
- transform: translate(-50%, 0) rotateZ(0deg);
- 50% {
- transform: translate(-50%, -2%) rotateZ(180deg);
- 100% {
- transform: translate(-50%, 0%) rotateZ(360deg);
- .monitor-active {
- background-color: #22bc76;
- color: #22bc76;
- .flex-content-right {
- padding: 0 15px 15px 7.5px;
- .task {
- height: 45px;
- .task-item {
- .task-item-box {
- width: 45px;
- z-index: 0;
- box-shadow: 0 10px 12px 0 rgba(0, 0, 0, 0.3);
- z-index: -2;
- left: -50%;
- top: -50%;
- width: 200%;
- height: 200%;
- background-repeat: no-repeat;
- background-size: 50% 50%, 50% 50%;
- background-position: 0 0, 100% 0, 100% 100%, 0 100%;
- background-image: linear-gradient(#19d4ae, #19d4ae), linear-gradient(#5ab1ef, #5ab1ef), linear-gradient(#fa6e86, #fa6e86),
- linear-gradient(#ffb980, #ffb980);
- animation: rotate 2s linear infinite;
- z-index: -1;
- left: 1px;
- top: 1px;
- width: calc(100% - 2px);
- height: calc(100% - 2px);
- .task-item-value {
- .task-item-label {
- .task1 {
- background: #5492be;
- .task2 {
- background: #43a177;
- .task3 {
- background: #a76077;
- .task-first-item {
- .task-first {
- .progress {
- .progress-item {
- .progress-box {
- margin-left: 10px;
- ::v-deep(.el-progress__text) {
- font-size: 12px !important;
- ::v-deep(.el-progress-bar__outer) {
- background-color: rgba(0, 0, 0, 0.1) !important;
- ::v-deep(.el-progress-bar) {
- margin-right: -22px !important;
@@ -1,59 +0,0 @@
-/**
- * sky 天气
- * @returns 返回模拟数据
-export const skyList = [
- v1: '时间',
- v2: '天气',
- v3: '温度',
- v5: '降水',
- v7: '风力',
- type: 'title',
- v1: '今天',
- v2: 'ele-Sunny',
- v3: '20°/26°',
- v5: '50%',
- v7: '13m/s',
- v1: '明天',
- v2: 'ele-Lightning',
-];
- * 当前设置状态
-export const dBtnList = [
- v2: '阳光玫瑰种植',
- v3: '126天',
- v4: '设备在线',
- * 当前设备监测
-export const chartData4List = [
- label: '温度',
- label: '光照',
- label: '湿度',
- label: '风力',
@@ -1,106 +0,0 @@
- <div class="big-data-up mb15">
- <div class="up-left">
- <i class="el-icon-time mr5"></i>
- <span>{{ time.txt }}</span>
- <div class="up-center">
- <span>智慧农业系统平台</span>
-import { reactive, toRefs, onBeforeMount, onUnmounted, defineComponent } from 'vue';
-import { formatDate } from '/@/utils/formatTime';
- name: 'chartHead',
- const state = reactive({
- time: {
- txt: '',
- fun: 0,
- });
- // 初始化时间
- const initTime = () => {
- state.time.txt = formatDate(new Date(), 'YYYY-mm-dd HH:MM:SS WWW QQQQ');
- state.time.fun = window.setInterval(() => {
- }, 1000);
- };
- // 页面加载前
- onBeforeMount(() => {
- initTime();
- // 页面卸载时
- onUnmounted(() => {
- window.clearInterval(state.time.fun);
- ...toRefs(state),
-});
-.big-data-up {
- height: 55px;
- padding: 0 15px;
- .up-left {
- .up-center {
- font-size: 18px;
- letter-spacing: 5px;
- background-image: -webkit-linear-gradient(
- left,
- var(--el-color-primary),
- var(--el-color-primary-light-1) 25%,
- var(--el-color-primary) 50%,
- var(--el-color-primary-light-1) 75%,
- var(--el-color-primary)
- );
- -webkit-text-fill-color: transparent;
- -webkit-background-clip: text;
- background-clip: text;
- background-size: 200% 100%;
- -webkit-animation: masked-animation-data-v-b02d8052 4s linear infinite;
- animation: masked-animation-data-v-b02d8052 4s linear infinite;
- -webkit-box-reflect: below -2px -webkit-gradient(linear, left top, left bottom, from(transparent), to(hsla(0, 0%, 100%, 0.1)));
- @keyframes masked-animation {
- background-position: 0 0;
- background-position: -100% 0;
- width: 250px;
- bottom: -15px;
- border: 1px transparent solid;
- border-image: linear-gradient(to right, var(--el-color-primary-light-9), var(--el-color-primary)) 1 10;
@@ -1,487 +0,0 @@
- <div class="chart-scrollbar layout-view-bg-white" :style="{ height: `calc(100vh - ${initTagViewHeight}` }">
- <div class="chart-warp">
- <div class="chart-warp-top">
- <ChartHead />
- <div class="chart-warp-bottom">
- <!-- 左边 -->
- <div class="big-data-down-left">
- <div class="flex-warp-item">
- <div class="flex-warp-item-box">
- <div class="flex-title">天气预报</div>
- <div class="flex-content">
- <div class="sky">
- <SvgIcon name="ele-Sunny" class="sky-left" />
- <div class="sky-center">
- <div class="mb2">
- <span>多云转晴</span>
- <span>东南风</span>
- <span class="span ml5">良</span>
- <div class="sky-right">
- <span>25</span>
- <span>°C</span>
- <div class="sky-dd">
- <div class="sky-dl" v-for="(v, k) in skyList" :key="k" :class="{ 'sky-dl-first': k === 1 }">
- <div>{{ v.v1 }}</div>
- <div v-if="v.type === 'title'">{{ v.v2 }}</div>
- <div v-else>
- <SvgIcon :name="v.v2" />
- <div>{{ v.v3 }}</div>
- <div class="tip">{{ v.v5 }}</div>
- <div>{{ v.v7 }}</div>
- <div class="flex-title">当前设备状态</div>
- <div class="flex-content flex-content-overflow">
- <div class="d-states">
- <div class="d-states-item">
- <SvgIcon name="ele-Odometer" class="i-bg1" />
- <div class="d-states-flex">
- <div class="d-states-item-label">园区设备数</div>
- <div class="d-states-item-value">99</div>
- <SvgIcon name="ele-FirstAidKit" class="i-bg2" />
- <div class="d-states-item-label">预警设备数</div>
- <div class="d-states-item-value">10</div>
- <SvgIcon name="ele-VideoPlay" class="i-bg3" />
- <div class="d-states-item-label">运行设备数</div>
- <div class="d-states-item-value">20</div>
- <div class="d-btn">
- <div class="d-btn-item" v-for="(v, k) in dBtnList" :key="k">
- <i class="d-btn-item-left el-icon-money"></i>
- <div class="d-btn-item-center">
- <div>{{ v.v2 }}|{{ v.v3 }}</div>
- <div class="d-btn-item-eight">{{ v.v4 }}</div>
- <div class="flex-title">近30天预警总数</div>
- <div style="height: 100%" ref="chartsWarningRef"></div>
- <!-- 中间 -->
- <div class="big-data-down-center">
- <div class="big-data-down-center-one">
- <div class="big-data-down-center-one-content">
- <div style="height: 100%" ref="chartsCenterOneRef"></div>
- <div class="big-data-down-center-two">
- <div class="flex-title">
- <span>当前设备监测</span>
- <span class="flex-title-small">单位:次</span>
- <div class="flex-content-left">
- <div class="monitor-item" v-for="(v, k) in chartData4List" :key="k">
- <div class="monitor-wave">
- <div class="monitor-z-index">
- <div class="monitor-item-label">{{ v.label }}</div>
- <div class="flex-content-right">
- <div style="height: 100%" ref="chartsMonitorRef"></div>
- <!-- 右边 -->
- <div class="big-data-down-right">
- <span>近7天产品追溯扫码统计</span>
- <div style="height: 100%" ref="chartsSevenDaysRef"></div>
- <div class="flex-title">当前任务统计</div>
- <div class="task">
- <div class="task-item task-first-item">
- <div class="task-item-value task-first">25</div>
- <div class="task-item-label">待办任务</div>
- <div class="task-item">
- <div class="task-item-box task1">
- <div class="task-item-value">12</div>
- <div class="task-item-label">施肥</div>
- <div class="task-item-box task2">
- <div class="task-item-value">3</div>
- <div class="task-item-label">施药</div>
- <div class="task-item-box task3">
- <div class="task-item-value">5</div>
- <div class="task-item-label">农事</div>
- <div class="progress">
- <div class="progress-item">
- <span>施肥率</span>
- <div class="progress-box">
- <el-progress :percentage="70" color="#43bdf0"></el-progress>
- <span>施药率</span>
- <el-progress :percentage="36" color="#43bdf0"></el-progress>
- <span>农事率</span>
- <el-progress :percentage="91" color="#43bdf0"></el-progress>
- <span>近7天投入品记录</span>
- <span class="flex-title-small">单位:件</span>
- <div style="height: 100%" ref="chartsInvestmentRef"></div>
-import { toRefs, reactive, computed, onMounted, getCurrentInstance, watch, nextTick, onActivated, defineComponent } from 'vue';
-import { useStore } from '/@/store/index';
-import ChartHead from './head.vue';
-import * as echarts from 'echarts';
-import 'echarts-wordcloud';
-import { skyList, dBtnList, chartData4List } from './chart';
- components: { ChartHead },
- const { proxy } = getCurrentInstance() as any;
- const store = useStore();
- skyList,
- dBtnList,
- chartData4List,
- myCharts: [],
- // 设置主内容的高度
- const initTagViewHeight = computed(() => {
- let { isTagsview } = store.state.themeConfig.themeConfig;
- let { isTagsViewCurrenFull } = store.state.tagsViewRoutes;
- if (isTagsViewCurrenFull) {
- return `30px`;
- if (isTagsview) return `114px`;
- else return `80px`;
- // 初始化中间图表1
- const initChartsCenterOne = () => {
- const myChart = echarts.init(proxy.$refs.chartsCenterOneRef);
- const option = {
- grid: {
- top: 15,
- right: 15,
- bottom: 20,
- left: 30,
- tooltip: {},
- series: [
- type: 'wordCloud',
- sizeRange: [12, 40],
- rotationRange: [0, 0],
- rotationStep: 45,
- gridSize: Math.random() * 20 + 5,
- shape: 'circle',
- width: '100%',
- height: '100%',
- textStyle: {
- fontFamily: 'sans-serif',
- fontWeight: 'bold',
- color: function () {
- return `rgb(${[Math.round(Math.random() * 160), Math.round(Math.random() * 160), Math.round(Math.random() * 160)].join(',')})`;
- data: [
- { name: 'vue-next-admin', value: 520 },
- { name: 'lyt', value: 520 },
- { name: 'next-admin', value: 500 },
- { name: '更名', value: 420 },
- { name: '智慧农业', value: 520 },
- { name: '男神', value: 2.64 },
- { name: '好身材', value: 4.03 },
- { name: '校草', value: 24.95 },
- { name: '酷', value: 4.04 },
- { name: '时尚', value: 5.27 },
- { name: '阳光活力', value: 5.8 },
- { name: '初恋', value: 3.09 },
- { name: '英俊潇洒', value: 24.71 },
- { name: '霸气', value: 6.33 },
- { name: '腼腆', value: 2.55 },
- { name: '蠢萌', value: 3.88 },
- { name: '青春', value: 8.04 },
- { name: '网红', value: 5.87 },
- { name: '萌', value: 6.97 },
- { name: '认真', value: 2.53 },
- { name: '古典', value: 2.49 },
- { name: '温柔', value: 3.91 },
- { name: '有个性', value: 3.25 },
- { name: '可爱', value: 9.93 },
- { name: '幽默诙谐', value: 3.65 },
- myChart.setOption(option);
- (<any>state.myCharts).push(myChart);
- // 初始化近7天产品追溯扫码统计
- const initChartsSevenDays = () => {
- const myChart = echarts.init(proxy.$refs.chartsSevenDaysRef);
- tooltip: {
- trigger: 'axis',
- xAxis: {
- type: 'category',
- boundaryGap: false,
- data: ['1天', '2天', '3天', '4天', '5天', '6天', '7天'],
- yAxis: {
- type: 'value',
- name: '邮件营销',
- type: 'line',
- stack: '总量',
- data: [12, 32, 11, 34, 90, 23, 21],
- name: '联盟广告',
- data: [22, 82, 91, 24, 90, 30, 30],
- name: '视频广告',
- data: [50, 32, 18, 14, 90, 30, 50],
- // 初始化近30天预警总数
- const initChartsWarning = () => {
- const myChart = echarts.init(proxy.$refs.chartsWarningRef);
- top: 50,
- right: 20,
- bottom: 30,
- trigger: 'item',
- name: '面积模式',
- type: 'pie',
- radius: [20, 50],
- center: ['50%', '50%'],
- roseType: 'area',
- itemStyle: {
- borderRadius: 8,
- { value: 40, name: '监测设备预警' },
- { value: 38, name: '天气预警' },
- { value: 32, name: '任务预警' },
- { value: 30, name: '病虫害预警' },
- // 初始化当前设备监测
- const initChartsMonitor = () => {
- const myChart = echarts.init(proxy.$refs.chartsMonitorRef);
- data: ['02:00', '04:00', '06:00', '08:00', '10:00', '12:00', '14:00'],
- color: '#289df5',
- borderColor: '#289df5',
- areaStyle: {
- type: 'default',
- opacity: 0.1,
- data: [20, 32, 31, 34, 12, 13, 20],
- areaStyle: {},
- // 初始化近7天投入品记录
- const initChartsInvestment = () => {
- const myChart = echarts.init(proxy.$refs.chartsInvestmentRef);
- data: [10, 20, 15, 80, 70, 11, 30],
- type: 'bar',
- // 批量设置 echarts resize
- const initEchartsResizeFun = () => {
- nextTick(() => {
- for (let i = 0; i < state.myCharts.length; i++) {
- (<any>state.myCharts[i]).resize();
- const initEchartsResize = () => {
- window.addEventListener('resize', initEchartsResizeFun);
- // 页面加载时
- onMounted(() => {
- initChartsCenterOne();
- initChartsSevenDays();
- initChartsWarning();
- initChartsMonitor();
- initChartsInvestment();
- initEchartsResize();
- // 由于页面缓存原因,keep-alive
- onActivated(() => {
- initEchartsResizeFun();
- // 监听 vuex 中的 tagsview 开启全屏变化,重新 resize 图表,防止不出现/大小不变等
- watch(
- () => store.state.tagsViewRoutes.isTagsViewCurrenFull,
- () => {
- initTagViewHeight,
-@import './chart.scss';
@@ -1,39 +0,0 @@
- <div id="printRref">
- <el-card shadow="hover" header="复制剪切演示">
- <el-alert
- title="感谢优秀的 `vue-clipboard3`,项目地址:https://github.com/JamieCurnow/vue-clipboard3`"
- type="success"
- :closable="false"
- class="mb15"
- ></el-alert>
- <el-input placeholder="请输入内容" v-model="copyVal">
- <template #append>
- <el-button @click="copyText(copyVal)">复制链接</el-button>
- </el-input>
- <el-input placeholder="先点击上方 `复制链接` 按钮,然后 `Ctrl + V` 进行粘贴! " v-model="shearVal" class="mt15"> </el-input>
-import { reactive, toRefs, onMounted, defineComponent } from 'vue';
-import commonFunction from '/@/utils/commonFunction';
- const { copyText } = commonFunction();
- copyVal: 'https://gitee.com/lyt-top/vue-next-admin',
- shearVal: '',
- onMounted(() => {});
- copyText,
@@ -1,161 +0,0 @@
- <div>
- <el-card shadow="hover" header="数字滚动演示">
- title="感谢优秀的 `countup.js`,项目地址:https://github.com/inorganik/countUp.js"
- <el-col :sm="6" class="mb15" v-for="(v, k) in topCardItemList" :key="k">
- <div class="countup-card-item countup-card-item-box" :style="{ background: `var(${v.color})` }">
- <div class="countup-card-item-flex">
- <div class="countup-card-item-title pb3">{{ v.title }}</div>
- <div class="countup-card-item-title-num pb6" :id="`titleNum${k + 1}`"></div>
- <div class="countup-card-item-tip pb3">{{ v.tip }}</div>
- <div class="countup-card-item-tip-num" :id="`tipNum${k + 1}`"></div>
- <i :class="v.icon" :style="{ color: v.iconColor }"></i>
- </el-col>
- <div class="flex-warp">
- <el-button type="primary" size="default" @click="refreshCurrent">
- <el-icon>
- <ele-RefreshRight />
- </el-icon>
- 重置/刷新数值
- </el-button>
-import { reactive, toRefs, onMounted, nextTick, defineComponent } from 'vue';
-import { CountUp } from 'countup.js';
- topCardItemList: [
- title: '今日访问人数',
- titleNum: '123',
- tip: '在场人数',
- tipNum: '911',
- color: '--el-color-primary',
- iconColor: '#ffcb47',
- icon: 'iconfont icon-jinridaiban',
- title: '实验室总数',
- tip: '使用中',
- tipNum: '611',
- color: '--el-color-success',
- iconColor: '#70cf41',
- icon: 'iconfont icon-AIshiyanshi',
- title: '申请人数(月)',
- tip: '通过人数',
- color: '--el-color-warning',
- iconColor: '#dfae64',
- icon: 'iconfont icon-shenqingkaiban',
- title: '销售情况',
- tip: '销售数',
- color: '--el-color-danger',
- iconColor: '#e56565',
- // 初始化数字滚动
- const initNumCountUp = () => {
- new CountUp('titleNum1', Math.random() * 10000).start();
- new CountUp('titleNum2', Math.random() * 10000).start();
- new CountUp('titleNum3', Math.random() * 10000).start();
- new CountUp('titleNum4', Math.random() * 10000).start();
- new CountUp('tipNum1', Math.random() * 1000).start();
- new CountUp('tipNum2', Math.random() * 1000).start();
- new CountUp('tipNum3', Math.random() * 1000).start();
- new CountUp('tipNum4', Math.random() * 1000).start();
- // 重置/刷新数值
- const refreshCurrent = () => {
- initNumCountUp();
- refreshCurrent,
-.countup-card-item {
- height: 103px;
- background: var(--el-text-color-secondary);
-.countup-card-item-box {
- right: 0px !important;
- bottom: 0px !important;
- right: -10px;
- bottom: -10px;
- font-size: 70px;
- transform: rotate(-30deg);
- .countup-card-item-flex {
- padding: 0 20px;
- .countup-card-item-title,
- .countup-card-item-tip {
- font-size: 13px;
- .countup-card-item-title-num {
- .countup-card-item-tip-num {
@@ -1,62 +0,0 @@
- <div class="croppers-container">
- <el-card shadow="hover" header="cropper 图片裁剪">
- title="感谢优秀的 `cropperjs`,项目地址:https://github.com/fengyuanchen/cropperjs"
- <div class="cropper-img-warp">
- <div class="mb15 mt15">
- <img class="cropper-img" :src="cropperImg" />
- <el-button type="primary" size="default" @click="onCropperDialogOpen">
- <ele-Crop />
- 更换头像
- <CropperDialog ref="cropperDialogRef" />
-import { ref, toRefs, reactive, defineComponent } from 'vue';
-import CropperDialog from '/@/components/cropper/index.vue';
- components: { CropperDialog },
- const cropperDialogRef = ref();
- cropperImg: 'https://img1.baidu.com/it/u=2813520958,2218166536&fm=26&fmt=auto&gp=0.jpg',
- // 打开裁剪弹窗
- const onCropperDialogOpen = () => {
- cropperDialogRef.value.openDialog(state.cropperImg);
- cropperDialogRef,
- onCropperDialogOpen,
-.croppers-container {
- .cropper-img-warp {
- .cropper-img {
- margin: auto;
- width: 150px;
- height: 150px;
@@ -1,135 +0,0 @@
- <div :style="{ height: `calc(100vh - ${initTagViewHeight}` }">
- <div class="layout-view-bg-white">
- <div id="echartsMap" style="height: 100%"></div>
-import { toRefs, reactive, computed, onMounted, defineComponent } from 'vue';
-import 'echarts/extension/bmap/bmap';
-import { echartsMapList, echartsMapData } from './mock';
- const state: any = reactive({
- echartsMapList,
- echartsMapData,
- // echartsMap 将坐标信息和对应物理量的值合在一起
- const convertData = (data: any) => {
- let res = [];
- for (let i = 0; i < data.length; i++) {
- let geoCoord = state.echartsMapData[data[i].name];
- if (geoCoord) {
- res.push({
- name: data[i].name,
- value: geoCoord.concat(data[i].value),
- return res;
- // 初始化 echartsMap
- const initEchartsMap = () => {
- const myChart = echarts.init(<HTMLElement>document.getElementById('echartsMap'));
- color: ['#9a60b4', '#ea7ccc'],
- bmap: {
- center: [104.114129, 37.550339],
- zoom: 5,
- roam: true,
- mapStyle: {},
- name: 'pm2.5',
- type: 'scatter',
- coordinateSystem: 'bmap',
- data: convertData(state.echartsMapList),
- symbolSize: function (val: any) {
- return val[2] / 10;
- encode: {
- value: 2,
- label: {
- formatter: '{b}',
- position: 'right',
- show: false,
- emphasis: {
- show: true,
- name: 'Top 5',
- type: 'effectScatter',
- data: convertData(
- state.echartsMapList
- .sort(function (a: any, b: any) {
- return b.value - a.value;
- .slice(0, 6)
- ),
- showEffectOn: 'render',
- rippleEffect: {
- brushType: 'stroke',
- hoverAnimation: true,
- shadowBlur: 10,
- shadowColor: '#333',
- zlevel: 1,
- window.addEventListener('resize', () => {
- myChart.resize();
- initEchartsMap();
@@ -1,387 +0,0 @@
-// 地图模拟数据
-export const echartsMapList = [
- { name: '海门', value: 9 },
- { name: '鄂尔多斯', value: 12 },
- { name: '招远', value: 12 },
- { name: '舟山', value: 12 },
- { name: '齐齐哈尔', value: 14 },
- { name: '盐城', value: 15 },
- { name: '赤峰', value: 16 },
- { name: '青岛', value: 18 },
- { name: '乳山', value: 18 },
- { name: '金昌', value: 19 },
- { name: '泉州', value: 21 },
- { name: '莱西', value: 21 },
- { name: '日照', value: 21 },
- { name: '胶南', value: 22 },
- { name: '南通', value: 23 },
- { name: '拉萨', value: 24 },
- { name: '云浮', value: 24 },
- { name: '梅州', value: 25 },
- { name: '文登', value: 25 },
- { name: '上海', value: 25 },
- { name: '攀枝花', value: 25 },
- { name: '威海', value: 25 },
- { name: '承德', value: 25 },
- { name: '厦门', value: 26 },
- { name: '汕尾', value: 26 },
- { name: '潮州', value: 26 },
- { name: '丹东', value: 27 },
- { name: '太仓', value: 27 },
- { name: '曲靖', value: 27 },
- { name: '烟台', value: 28 },
- { name: '福州', value: 29 },
- { name: '瓦房店', value: 30 },
- { name: '即墨', value: 30 },
- { name: '抚顺', value: 31 },
- { name: '玉溪', value: 31 },
- { name: '张家口', value: 31 },
- { name: '阳泉', value: 31 },
- { name: '莱州', value: 32 },
- { name: '湖州', value: 32 },
- { name: '汕头', value: 32 },
- { name: '昆山', value: 33 },
- { name: '宁波', value: 33 },
- { name: '湛江', value: 33 },
- { name: '揭阳', value: 34 },
- { name: '荣成', value: 34 },
- { name: '连云港', value: 35 },
- { name: '葫芦岛', value: 35 },
- { name: '常熟', value: 36 },
- { name: '东莞', value: 36 },
- { name: '河源', value: 36 },
- { name: '淮安', value: 36 },
- { name: '泰州', value: 36 },
- { name: '南宁', value: 37 },
- { name: '营口', value: 37 },
- { name: '惠州', value: 37 },
- { name: '江阴', value: 37 },
- { name: '蓬莱', value: 37 },
- { name: '韶关', value: 38 },
- { name: '嘉峪关', value: 38 },
- { name: '广州', value: 38 },
- { name: '延安', value: 38 },
- { name: '太原', value: 39 },
- { name: '清远', value: 39 },
- { name: '中山', value: 39 },
- { name: '昆明', value: 39 },
- { name: '寿光', value: 40 },
- { name: '盘锦', value: 40 },
- { name: '长治', value: 41 },
- { name: '深圳', value: 360 },
- { name: '珠海', value: 42 },
- { name: '宿迁', value: 43 },
- { name: '咸阳', value: 43 },
- { name: '铜川', value: 44 },
- { name: '平度', value: 44 },
- { name: '佛山', value: 44 },
- { name: '海口', value: 44 },
- { name: '江门', value: 45 },
- { name: '章丘', value: 45 },
- { name: '肇庆', value: 46 },
- { name: '大连', value: 47 },
- { name: '临汾', value: 47 },
- { name: '吴江', value: 47 },
- { name: '石嘴山', value: 49 },
- { name: '沈阳', value: 50 },
- { name: '苏州', value: 50 },
- { name: '茂名', value: 50 },
- { name: '嘉兴', value: 51 },
- { name: '长春', value: 51 },
- { name: '胶州', value: 52 },
- { name: '银川', value: 52 },
- { name: '张家港', value: 52 },
- { name: '三门峡', value: 53 },
- { name: '锦州', value: 54 },
- { name: '南昌', value: 54 },
- { name: '柳州', value: 54 },
- { name: '三亚', value: 54 },
- { name: '自贡', value: 56 },
- { name: '吉林', value: 56 },
- { name: '阳江', value: 57 },
- { name: '泸州', value: 57 },
- { name: '西宁', value: 57 },
- { name: '宜宾', value: 58 },
- { name: '呼和浩特', value: 58 },
- { name: '成都', value: 58 },
- { name: '大同', value: 58 },
- { name: '镇江', value: 59 },
- { name: '桂林', value: 59 },
- { name: '张家界', value: 59 },
- { name: '宜兴', value: 59 },
- { name: '北海', value: 60 },
- { name: '西安', value: 61 },
- { name: '金坛', value: 62 },
- { name: '东营', value: 62 },
- { name: '牡丹江', value: 63 },
- { name: '遵义', value: 63 },
- { name: '绍兴', value: 63 },
- { name: '扬州', value: 64 },
- { name: '常州', value: 64 },
- { name: '潍坊', value: 65 },
- { name: '重庆', value: 66 },
- { name: '台州', value: 67 },
- { name: '南京', value: 67 },
- { name: '滨州', value: 70 },
- { name: '贵阳', value: 71 },
- { name: '无锡', value: 71 },
- { name: '本溪', value: 71 },
- { name: '克拉玛依', value: 72 },
- { name: '渭南', value: 72 },
- { name: '马鞍山', value: 72 },
- { name: '宝鸡', value: 72 },
- { name: '焦作', value: 75 },
- { name: '句容', value: 75 },
- { name: '北京', value: 79 },
- { name: '徐州', value: 79 },
- { name: '衡水', value: 80 },
- { name: '包头', value: 80 },
- { name: '绵阳', value: 80 },
- { name: '乌鲁木齐', value: 84 },
- { name: '枣庄', value: 84 },
- { name: '杭州', value: 84 },
- { name: '淄博', value: 85 },
- { name: '鞍山', value: 86 },
- { name: '溧阳', value: 86 },
- { name: '库尔勒', value: 86 },
- { name: '安阳', value: 90 },
- { name: '开封', value: 90 },
- { name: '济南', value: 92 },
- { name: '德阳', value: 93 },
- { name: '温州', value: 95 },
- { name: '九江', value: 96 },
- { name: '邯郸', value: 98 },
- { name: '临安', value: 99 },
- { name: '兰州', value: 99 },
- { name: '沧州', value: 100 },
- { name: '临沂', value: 103 },
- { name: '南充', value: 104 },
- { name: '天津', value: 105 },
- { name: '富阳', value: 106 },
- { name: '泰安', value: 112 },
- { name: '诸暨', value: 112 },
- { name: '郑州', value: 113 },
- { name: '哈尔滨', value: 114 },
- { name: '聊城', value: 116 },
- { name: '芜湖', value: 117 },
- { name: '唐山', value: 119 },
- { name: '平顶山', value: 119 },
- { name: '邢台', value: 119 },
- { name: '德州', value: 120 },
- { name: '济宁', value: 120 },
- { name: '荆州', value: 127 },
- { name: '宜昌', value: 130 },
- { name: '义乌', value: 132 },
- { name: '丽水', value: 133 },
- { name: '洛阳', value: 134 },
- { name: '秦皇岛', value: 136 },
- { name: '株洲', value: 143 },
- { name: '石家庄', value: 147 },
- { name: '莱芜', value: 148 },
- { name: '常德', value: 152 },
- { name: '保定', value: 153 },
- { name: '湘潭', value: 154 },
- { name: '金华', value: 157 },
- { name: '岳阳', value: 169 },
- { name: '长沙', value: 175 },
- { name: '衢州', value: 177 },
- { name: '廊坊', value: 93 },
- { name: '菏泽', value: 194 },
- { name: '合肥', value: 229 },
- { name: '武汉', value: 273 },
- { name: '大庆', value: 279 },
-// 地图经纬度数据
-export const echartsMapData = {
- 海门: [121.15, 31.89],
- 鄂尔多斯: [109.781327, 39.608266],
- 招远: [120.38, 37.35],
- 舟山: [122.207216, 29.985295],
- 齐齐哈尔: [123.97, 47.33],
- 盐城: [120.13, 33.38],
- 赤峰: [118.87, 42.28],
- 青岛: [120.33, 36.07],
- 乳山: [121.52, 36.89],
- 金昌: [102.188043, 38.520089],
- 泉州: [118.58, 24.93],
- 莱西: [120.53, 36.86],
- 日照: [119.46, 35.42],
- 胶南: [119.97, 35.88],
- 南通: [121.05, 32.08],
- 拉萨: [91.11, 29.97],
- 云浮: [112.02, 22.93],
- 梅州: [116.1, 24.55],
- 文登: [122.05, 37.2],
- 上海: [121.48, 31.22],
- 攀枝花: [101.718637, 26.582347],
- 威海: [122.1, 37.5],
- 承德: [117.93, 40.97],
- 厦门: [118.1, 24.46],
- 汕尾: [115.375279, 22.786211],
- 潮州: [116.63, 23.68],
- 丹东: [124.37, 40.13],
- 太仓: [121.1, 31.45],
- 曲靖: [103.79, 25.51],
- 烟台: [121.39, 37.52],
- 福州: [119.3, 26.08],
- 瓦房店: [121.979603, 39.627114],
- 即墨: [120.45, 36.38],
- 抚顺: [123.97, 41.97],
- 玉溪: [102.52, 24.35],
- 张家口: [114.87, 40.82],
- 阳泉: [113.57, 37.85],
- 莱州: [119.942327, 37.177017],
- 湖州: [120.1, 30.86],
- 汕头: [116.69, 23.39],
- 昆山: [120.95, 31.39],
- 宁波: [121.56, 29.86],
- 湛江: [110.359377, 21.270708],
- 揭阳: [116.35, 23.55],
- 荣成: [122.41, 37.16],
- 连云港: [119.16, 34.59],
- 葫芦岛: [120.836932, 40.711052],
- 常熟: [120.74, 31.64],
- 东莞: [113.75, 23.04],
- 河源: [114.68, 23.73],
- 淮安: [119.15, 33.5],
- 泰州: [119.9, 32.49],
- 南宁: [108.33, 22.84],
- 营口: [122.18, 40.65],
- 惠州: [114.4, 23.09],
- 江阴: [120.26, 31.91],
- 蓬莱: [120.75, 37.8],
- 韶关: [113.62, 24.84],
- 嘉峪关: [98.289152, 39.77313],
- 广州: [113.23, 23.16],
- 延安: [109.47, 36.6],
- 太原: [112.53, 37.87],
- 清远: [113.01, 23.7],
- 中山: [113.38, 22.52],
- 昆明: [102.73, 25.04],
- 寿光: [118.73, 36.86],
- 盘锦: [122.070714, 41.119997],
- 长治: [113.08, 36.18],
- 深圳: [114.07, 22.62],
- 珠海: [113.52, 22.3],
- 宿迁: [118.3, 33.96],
- 咸阳: [108.72, 34.36],
- 铜川: [109.11, 35.09],
- 平度: [119.97, 36.77],
- 佛山: [113.11, 23.05],
- 海口: [110.35, 20.02],
- 江门: [113.06, 22.61],
- 章丘: [117.53, 36.72],
- 肇庆: [112.44, 23.05],
- 大连: [121.62, 38.92],
- 临汾: [111.5, 36.08],
- 吴江: [120.63, 31.16],
- 石嘴山: [106.39, 39.04],
- 沈阳: [123.38, 41.8],
- 苏州: [120.62, 31.32],
- 茂名: [110.88, 21.68],
- 嘉兴: [120.76, 30.77],
- 长春: [125.35, 43.88],
- 胶州: [120.03336, 36.264622],
- 银川: [106.27, 38.47],
- 张家港: [120.555821, 31.875428],
- 三门峡: [111.19, 34.76],
- 锦州: [121.15, 41.13],
- 南昌: [115.89, 28.68],
- 柳州: [109.4, 24.33],
- 三亚: [109.511909, 18.252847],
- 自贡: [104.778442, 29.33903],
- 吉林: [126.57, 43.87],
- 阳江: [111.95, 21.85],
- 泸州: [105.39, 28.91],
- 西宁: [101.74, 36.56],
- 宜宾: [104.56, 29.77],
- 呼和浩特: [111.65, 40.82],
- 成都: [104.06, 30.67],
- 大同: [113.3, 40.12],
- 镇江: [119.44, 32.2],
- 桂林: [110.28, 25.29],
- 张家界: [110.479191, 29.117096],
- 宜兴: [119.82, 31.36],
- 北海: [109.12, 21.49],
- 西安: [108.95, 34.27],
- 金坛: [119.56, 31.74],
- 东营: [118.49, 37.46],
- 牡丹江: [129.58, 44.6],
- 遵义: [106.9, 27.7],
- 绍兴: [120.58, 30.01],
- 扬州: [119.42, 32.39],
- 常州: [119.95, 31.79],
- 潍坊: [119.1, 36.62],
- 重庆: [106.54, 29.59],
- 台州: [121.420757, 28.656386],
- 南京: [118.78, 32.04],
- 滨州: [118.03, 37.36],
- 贵阳: [106.71, 26.57],
- 无锡: [120.29, 31.59],
- 本溪: [123.73, 41.3],
- 克拉玛依: [84.77, 45.59],
- 渭南: [109.5, 34.52],
- 马鞍山: [118.48, 31.56],
- 宝鸡: [107.15, 34.38],
- 焦作: [113.21, 35.24],
- 句容: [119.16, 31.95],
- 北京: [116.46, 39.92],
- 徐州: [117.2, 34.26],
- 衡水: [115.72, 37.72],
- 包头: [110, 40.58],
- 绵阳: [104.73, 31.48],
- 乌鲁木齐: [87.68, 43.77],
- 枣庄: [117.57, 34.86],
- 杭州: [120.19, 30.26],
- 淄博: [118.05, 36.78],
- 鞍山: [122.85, 41.12],
- 溧阳: [119.48, 31.43],
- 库尔勒: [86.06, 41.68],
- 安阳: [114.35, 36.1],
- 开封: [114.35, 34.79],
- 济南: [117, 36.65],
- 德阳: [104.37, 31.13],
- 温州: [120.65, 28.01],
- 九江: [115.97, 29.71],
- 邯郸: [114.47, 36.6],
- 临安: [119.72, 30.23],
- 兰州: [103.73, 36.03],
- 沧州: [116.83, 38.33],
- 临沂: [118.35, 35.05],
- 南充: [106.110698, 30.837793],
- 天津: [117.2, 39.13],
- 富阳: [119.95, 30.07],
- 泰安: [117.13, 36.18],
- 诸暨: [120.23, 29.71],
- 郑州: [113.65, 34.76],
- 哈尔滨: [126.63, 45.75],
- 聊城: [115.97, 36.45],
- 芜湖: [118.38, 31.33],
- 唐山: [118.02, 39.63],
- 平顶山: [113.29, 33.75],
- 邢台: [114.48, 37.05],
- 德州: [116.29, 37.45],
- 济宁: [116.59, 35.38],
- 荆州: [112.239741, 30.335165],
- 宜昌: [111.3, 30.7],
- 义乌: [120.06, 29.32],
- 丽水: [119.92, 28.45],
- 洛阳: [112.44, 34.7],
- 秦皇岛: [119.57, 39.95],
- 株洲: [113.16, 27.83],
- 石家庄: [114.48, 38.03],
- 莱芜: [117.67, 36.19],
- 常德: [111.69, 29.05],
- 保定: [115.48, 38.85],
- 湘潭: [112.91, 27.87],
- 金华: [119.64, 29.12],
- 岳阳: [113.09, 29.37],
- 长沙: [113, 28.21],
- 衢州: [118.88, 28.97],
- 廊坊: [116.7, 39.53],
- 菏泽: [115.480656, 35.23375],
- 合肥: [117.27, 31.86],
- 武汉: [114.31, 30.52],
- 大庆: [125.03, 46.58],
-};
@@ -1,61 +0,0 @@
- <div class="grid-layout-container">
- <el-card shadow="hover" header="vue-grid-layout 拖拽布局演示">
- title="感谢优秀的 `vue-grid-layout`,项目地址:https://github.com/jbaysolutions/vue-grid-layout"
- <grid-layout
- v-model:layout="layouts"
- :col-num="12"
- :row-height="30"
- :is-draggable="true"
- :is-resizable="true"
- :is-mirrored="false"
- :vertical-compact="true"
- :margin="[10, 10]"
- :use-css-transforms="true"
- <grid-item v-for="item in layouts" :x="item.x" :y="item.y" :w="item.w" :h="item.h" :i="item.i" :key="item.i">
- <div class="w100 h100 flex">
- <span class="flex-margin font14">{{ item.i }}</span>
- </grid-item>
- </grid-layout>
-import { toRefs, reactive, defineComponent } from 'vue';
- name: 'FunGridLayout',
- layouts: [
- { x: 0, y: 0, w: 2, h: 2, i: '0' },
- { x: 2, y: 0, w: 2, h: 4, i: '1' },
- { x: 4, y: 0, w: 2, h: 5, i: '2' },
- { x: 6, y: 0, w: 2, h: 3, i: '3' },
- { x: 8, y: 0, w: 2, h: 3, i: '4' },
- { x: 10, y: 0, w: 2, h: 3, i: '5' },
- { x: 0, y: 5, w: 2, h: 5, i: '6' },
-.grid-layout-container {
- .vue-grid-item {
- background: var(--el-color-primary);
@@ -1,41 +0,0 @@
- <div id="printRef">
- <el-card shadow="hover" header="打印演示">
- title="感谢优秀的 `print-js`,项目地址:https://github.com/crabbly/Print.js。请在打印弹窗 `更多设置` 中开启 `背景图形。`"
- <el-button @click="onPrintJs" size="default" type="primary">
- <SvgIcon name="iconfont icon-dayin" />
- 点击打印演示
-import { reactive, toRefs, defineComponent } from 'vue';
-import printJs from 'print-js';
- const state = reactive({});
- // 打印点击
- const onPrintJs = () => {
- printJs({
- printable: 'printRef',
- type: 'html',
- css: ['//at.alicdn.com/t/font_2298093_o73r8wjdhlg.css', '//unpkg.com/element-plus/dist/index.css'],
- scanStyles: false,
- style: `@media print{.mb15{margin-bottom:15px;}.el-button--small i.iconfont{font-size: 12px !important;margin-right: 5px;}}`,
- onPrintJs,
@@ -1,78 +0,0 @@
- <div class="qrcode-container">
- <el-card shadow="hover" header="qrcodejs2 二维码生成">
- title="感谢优秀的 `qrcodejs2`,项目地址:https://github.com/davidshimjs/qrcodejs"
- <div class="qrcode-img-warp">
- <div class="mb30 mt30 qrcode-img">
- <div class="qrcode" ref="qrcodeRef"></div>
- <el-button type="primary" size="default" @click="onInitQrcode">
- <ele-Refresh />
- 重新生成
-import { toRefs, reactive, onMounted, getCurrentInstance, defineComponent } from 'vue';
-import QRCode from 'qrcodejs2-fixes';
- qrcode: '',
- // 初始化生成二维码
- const initQrcode = () => {
- new QRCode(proxy.$refs.qrcodeRef, {
- text: `https://lyt-top.gitee.io/vue-next-admin-preview/#/login?t=${new Date().getTime()}`,
- width: 125,
- height: 125,
- colorDark: '#000000',
- colorLight: '#ffffff',
- // 重新生成
- const onInitQrcode = () => {
- proxy.$refs.qrcodeRef.innerHTML = '';
- initQrcode();
- onInitQrcode,
-.qrcode-container {
- .qrcode-img-warp {
- .qrcode-img {
- height: 125px;
- .qrcode {
- width: 125px;
@@ -1,53 +0,0 @@
- <div class="splitpanes-container">
- <el-card shadow="hover" header="splitpanes 窗格拆分器">
- title="感谢优秀的 `splitpanes`,项目地址:https://github.com/antoniandre/splitpanes"
- <splitpanes class="default-theme" @resize="paneSize = $event[0].size" style="height: 500px">
- <pane :size="32"> 1 </pane>
- <pane :size="36">
- <splitpanes class="default-theme" :horizontal="true">
- <pane :size="100"> 2 </pane>
- <pane :size="100"> 3 </pane>
- </splitpanes>
- </pane>
- <pane :size="32"> 4 </pane>
-import { Splitpanes, Pane } from 'splitpanes';
-import 'splitpanes/dist/splitpanes.css';
- components: { Splitpanes, Pane },
- paneSize: 50,
-.splitpanes__pane {
- color: var(--el-color-primary-light-5);
- background-color: var(--el-color-primary) !important;
@@ -1,117 +0,0 @@
- <div class="fun-tagsview">
- <NoticeBar
- text="已删除非当前页 tagsView 演示,后续有时间可以再加回来!,tagsview 支持多标签(参数不同)、单标签共用(参数不同)"
- background="#ecf5ff"
- color="#409eff"
- />
- <el-card shadow="hover" header="tagsView 当前页演示" class="mt15">
- <el-button type="primary" size="default" @click="refreshCurrentTagsView">
- 刷新当前页
- <el-button type="info" size="default" @click="closeCurrentTagsView">
- <ele-Close />
- 关闭当前页
- <el-button type="warning" size="default" @click="closeOtherTagsView">
- <ele-CircleClose />
- 关闭其它
- <el-button type="danger" size="default" @click="closeAllTagsView">
- <ele-FolderDelete />
- 全部关闭
- <el-button type="success" size="default" @click="openCurrenFullscreen">
- <ele-FullScreen />
- 当前页全屏
-import { getCurrentInstance, reactive, toRefs, defineComponent } from 'vue';
-import NoticeBar from '/@/components/noticeBar/index.vue';
-import { useRoute } from 'vue-router';
- components: { NoticeBar },
- const route = useRoute();
- // 0 刷新当前,1 关闭当前,2 关闭其它,3 关闭全部 4 当前页全屏
- // 1、刷新当前 tagsView
- const refreshCurrentTagsView = () => {
- proxy.mittBus.emit('onCurrentContextmenuClick', Object.assign({}, { contextMenuClickId: 0, ...route }));
- // 2、关闭当前 tagsView
- const closeCurrentTagsView = () => {
- proxy.mittBus.emit('onCurrentContextmenuClick', Object.assign({}, { contextMenuClickId: 1, ...route }));
- // 3、关闭其它 tagsView
- const closeOtherTagsView = () => {
- proxy.mittBus.emit('onCurrentContextmenuClick', Object.assign({}, { contextMenuClickId: 2, ...route }));
- // 4、关闭全部 tagsView
- const closeAllTagsView = () => {
- proxy.mittBus.emit('onCurrentContextmenuClick', Object.assign({}, { contextMenuClickId: 3, ...route }));
- // 5、开启当前页面全屏
- const openCurrenFullscreen = () => {
- proxy.mittBus.emit('onCurrentContextmenuClick', Object.assign({}, { contextMenuClickId: 4, ...route }));
- refreshCurrentTagsView,
- closeCurrentTagsView,
- closeOtherTagsView,
- closeAllTagsView,
- openCurrenFullscreen,
-.fun-tagsview {
- .fun-tagsview-from-item {
- ::v-deep(.el-form-item__content) {
- margin-left: 0 !important;
@@ -1,32 +0,0 @@
- <div class="editor-container">
- <el-card shadow="hover" header="wangeditor富文本编辑器">
- title="感谢优秀的 `wangeditor`,项目地址:https://github.com/wangeditor-team/wangEditor"
- <Editor :is-disable="false" v-model="editorVal" />
-import { toRefs, reactive, onMounted, defineComponent } from 'vue';
-import Editor from '/@/components/editor/index.vue';
- components: { Editor },
- editorVal: '',
@@ -1,163 +0,0 @@
- <div class="notice-bar-container">
- <el-card shadow="hover" header="滚动通知栏:默认">
- text="🎉🎉🔥基于vue3.x 、Typescript、vite、Element plus等,适配手机、平板、pc
- 的后台开源免费模板库(vue2.x请切换vue-prev-admin分支),仓库地址:https://gitee.com/lyt-top/vue-next-admin"
- <el-card shadow="hover" header="滚动通知栏:设置样式" class="mt15">
- leftIcon="iconfont icon-tongzhi2"
- rightIcon="ele-ArrowRight"
- <el-card shadow="hover" header="滚动通知栏:搭配 NoticeBar 和 Carousel 走马灯 组件可以实现垂直滚动的效果" class="mt15">
- <NoticeBar :scrollable="true">
- <el-carousel height="40px" direction="vertical" :autoplay="true" indicator-position="none" :interval="3000">
- <el-carousel-item v-for="v in noticeList" :key="v">{{ v }} </el-carousel-item>
- </el-carousel>
- </NoticeBar>
- <el-card shadow="hover" header="滚动通知栏:参数" class="mt15">
- <el-table :data="tableData" style="width: 100%">
- <el-table-column prop="a1" label="参数"> </el-table-column>
- <el-table-column prop="a2" label="说明"> </el-table-column>
- <el-table-column prop="a3" label="类型"> </el-table-column>
- <el-table-column prop="a4" label="可选值"> </el-table-column>
- <el-table-column prop="a5" label="默认值"> </el-table-column>
- </el-table>
- <el-card shadow="hover" header="图标选择器(宽度自动):事件" class="mt15">
- <el-table :data="tableData1" style="width: 100%">
- <el-table-column prop="a1" label="事件名称"> </el-table-column>
- <el-table-column prop="a4" label="回调参数"> </el-table-column>
- noticeList: [
- '🎉🎉🔥基于vue3.x 、Typescript、vite、Element plus等',
- '适配手机、平板、pc的后台开源免费模板库(vue2.x请切换vue-prev-admin分支)',
- '仓库地址:https://gitee.com/lyt-top/vue-next-admin',
- '演示地址:https://lyt-top.gitee.io/vue-next-admin-preview/#/login',
- tableData: [
- a1: 'mode',
- a2: '通知栏模式,用于右侧 icon 图标点击',
- a3: 'string',
- a4: 'closeable / link',
- a5: '',
- a1: 'text',
- a2: '通知文本内容,scrollable 为 false 时生效',
- a4: '',
- a1: 'color',
- a2: '通知文本颜色',
- a5: '#e6a23c',
- a1: 'background',
- a2: '通知背景色',
- a5: '#fdf6ec',
- a1: 'size',
- a2: '字体大小,单位px',
- a3: 'number / string',
- a5: '14',
- a1: 'height',
- a2: '通知栏高度,单位px',
- a5: '40',
- a1: 'delay',
- a2: '动画延迟时间 (s)',
- a5: '1',
- a1: 'speed',
- a2: '滚动速率 (px/s)',
- a5: '100',
- a1: 'scrollable',
- a2: '是否开启垂直滚动',
- a3: 'boolean',
- a4: 'true',
- a5: 'false',
- a1: 'leftIcon',
- a2: '自定义左侧图标',
- a1: 'rightIcon',
- a2: '自定义右侧图标',
- tableData1: [
- a1: 'close',
- a2: '通知栏模式(mode)closeable 时回调事件',
- a3: 'function',
- a1: 'link',
- a2: '通知栏模式(mode)link 时回调事件',
@@ -1,125 +0,0 @@
- <div class="selector-container">
- <el-card shadow="hover" header="图标选择器(宽度自动):">
- <IconSelector @get="onGetIcon" @clear="onClearIcon" v-model="modelIcon" />
- <el-card shadow="hover" header="图标选择器(宽度自动):参数" class="mt15">
-import IconSelector from '/@/components/iconSelector/index.vue';
- components: { IconSelector },
- modelIcon: '',
- a1: 'prepend',
- a2: '输入框前置内容,只能字体图标',
- a5: 'ele-Pointer',
- a1: 'placeholder',
- a2: '输入框占位文本',
- a5: '请输入内容搜索图标或者选择图标',
- a2: '尺寸',
- a4: 'large / default / small',
- a5: 'default',
- a1: 'title',
- a2: '弹窗标题',
- a5: '请选择图标',
- a1: 'type',
- a2: 'icon 图标类型',
- a4: 'ali / ele / awe / all',
- a5: 'ele',
- a1: 'disabled',
- a2: '禁用',
- a1: 'clearable',
- a2: '是否可清空',
- a4: 'false',
- a5: 'true',
- a1: 'emptyDescription',
- a2: '自定义空状态描述文字',
- a3: 'String',
- a5: '无相关图标',
- a1: 'get',
- a2: '获取当前点击的 icon 图标',
- a4: '(icon: string)',
- a1: 'clear',
- a2: '清空当前点击的 icon 图标',
- // 获取当前点击的 icon 图标
- const onGetIcon = (icon: string) => {
- console.log(icon);
- // 清空当前点击的 icon 图标
- const onClearIcon = (icon: string) => {
- onGetIcon,
- onClearIcon,
@@ -1,60 +0,0 @@
- <div class="svg-demo-container">
- <el-card shadow="hover" header="svgIcon:演示(支持本地svg)">
- <SvgIcon name="iconfont icon-shuju1" color="red" :size="30" />
- <SvgIcon name="ele-Trophy" color="var(--el-color-primary)" :size="30" />
- <SvgIcon name="fa fa-flag-checkered" color="#09f" :size="30" />
- <SvgIcon :name="logoMini" color="#09f" :size="30" />
- <el-card shadow="hover" header="svgIcon:参数" class="mt15">
-import logoMini from '/@/assets/logo-mini.svg';
- a1: 'name',
- a2: 'svg 图标组件名字 / svg 路径 url',
- a2: 'svg 大小',
- a3: 'number',
- a5: 14,
- a2: 'svg 颜色',
- logoMini,
@@ -1,20 +0,0 @@
- <el-input v-model="val" placeholder="menu11:请输入内容测试路由缓存"></el-input>
- val: '',
- <el-input v-model="val" placeholder="menu121:请输入内容测试路由缓存"></el-input>
- <el-input v-model="val" placeholder="menu122:请输入内容测试路由缓存"></el-input>
@@ -1,26 +0,0 @@
- <el-input v-model="val" placeholder="menu13:请输入内容测试路由缓存"></el-input>
-import { toRefs, reactive, onActivated, onMounted, defineComponent } from 'vue';
- console.log(2222);
- console.log(1111);
- <el-input v-model="val" placeholder="menu2:请输入内容测试路由缓存"></el-input>
@@ -1,86 +0,0 @@
- <div class="awesome-container">
- <el-card shadow="hover" :header="`fontawesome 字体图标(自动载入):${sheetsIconList.length - 24}个`">
- <el-row class="iconfont-row">
- <el-col :xs="12" :sm="8" :md="6" :lg="4" :xl="2" v-for="(v, k) in sheetsIconList" :key="k">
- <div class="iconfont-warp">
- <div class="flex-margin">
- <div class="iconfont-warp-value">
- <i :class="v" class="fa"></i>
- <div class="iconfont-warp-label mt10">{{ v }}</div>
-import initIconfont from '/@/utils/getStyleSheets';
- sheetsIconList: [],
- // 初始化获取 css 样式,这里使用fontawesome的图标(记得加上前缀 `fa`),其它第三方请自行做判断
- const initGetStyleSheets = () => {
- initIconfont.awe().then((res: any) => (state.sheetsIconList = res));
- initGetStyleSheets();
-.awesome-container {
- .iconfont-row {
- border-top: 1px solid var(--next-border-color-light);
- border-left: 1px solid var(--next-border-color-light);
- .iconfont-warp {
- border-right: 1px solid var(--next-border-color-light);
- border-bottom: 1px solid var(--next-border-color-light);
- height: 120px;
- transition: all 0.3s ease;
- box-shadow: 0 2px 12px var(--next-color-dark-hover);
- .iconfont-warp-value {
- .iconfont-warp-label {
- color: #606266;
- font-size: 32px;
- color: #99a9bf;
@@ -1,65 +0,0 @@
- <div class="drag-container">
- <el-card shadow="hover" header="拖动指令效果(v-drag)作用于 Dialog 对话框">
- <el-button type="primary" @click="dialogVisible = true" size="default">
- <ele-Pointer />
- 点击打开 Dialog
- <el-card shadow="hover" header="自定义div" class="mt15">
- <div class="drag-dom">
- <div class="drag-header">
- <el-button type="success" size="default" v-drag="['.drag-container .drag-dom', '.drag-container .drag-header']">
- 按住进行拖动测试
- <el-dialog v-model="dialogVisible" width="769px">
- <template #title>
- <div v-drag="['.drag-container .el-dialog', '.drag-container .el-dialog__header']">拖动指令效果(v-drag)</div>
- <p>鼠标放标题头进行 Dialog 对话框拖动</p>
- <template #footer>
- <span class="dialog-footer">
- <el-button @click="dialogVisible = false" size="default">取 消</el-button>
- <el-button type="primary" @click="dialogVisible = false" size="default">确 定</el-button>
- </span>
- </el-dialog>
- dialogVisible: false,
-.drag-container {
- .drag-dom {
- display: inline-block;
- .drag-header {
@@ -1,204 +0,0 @@
- <div class="dynamic-form-container">
- <el-card shadow="hover" header="动态复杂表单">
- <el-form :model="form" ref="formRulesOneRef" size="default" label-width="100px" class="mt35">
- <el-row :gutter="35">
- <el-col
- :xs="val.xs"
- :sm="val.sm"
- :md="val.md"
- :lg="val.md"
- :xl="val.xl"
- class="mb20"
- v-show="val.isShow"
- v-for="(val, key) in formData"
- :key="key"
- <template v-if="val.type !== ''">
- <el-form-item
- :label="val.label"
- :prop="val.prop"
- :rules="[{ required: val.required, message: `${val.label}不能为空`, trigger: val.type === 'input' ? 'blur' : 'change' }]"
- v-if="val.type !== ''"
- <el-input
- v-model="form[val.prop]"
- :placeholder="val.placeholder"
- clearable
- v-if="val.type === 'input'"
- style="width: 100%"
- :disabled="val.disabled"
- ></el-input>
- <el-date-picker
- type="date"
- v-else-if="val.type === 'date'"
- </el-date-picker>
- <el-select
- v-else-if="val.type === 'select'"
- <el-option v-for="item in val.options" :key="item.value" :label="item.label" :value="item.value"> </el-option>
- </el-select>
- type="textarea"
- v-if="val.type === 'textarea'"
- <template v-else>
- <el-row :gutter="35" v-for="(v, k) in form.list" :key="k">
- <el-col :xs="24" :sm="12" :md="8" :lg="8" :xl="6" class="mb20">
- <el-form-item label="年度" :prop="`list[${k}].year`" :rules="[{ required: true, message: `年度不能为空`, trigger: 'blur' }]">
- <template #label>
- <el-button type="primary" circle size="small" @click="onAddRow" v-if="k === 0">
- <ele-Plus />
- <el-button type="danger" circle size="small" @click="onDelRow(k)" v-else>
- <ele-Delete />
- <span class="ml10">年度</span>
- <el-input v-model="form.list[k].year" style="width: 100%" placeholder="请输入"> </el-input>
- <el-form-item label="月度" :prop="`list[${k}].month`" :rules="[{ required: true, message: `月度不能为空`, trigger: 'blur' }]">
- <el-input v-model="form.list[k].month" style="width: 100%" placeholder="请输入"> </el-input>
- <el-form-item label="日度" :prop="`list[${k}].day`" :rules="[{ required: true, message: `日度不能为空`, trigger: 'blur' }]">
- <el-input v-model="form.list[k].day" style="width: 100%" placeholder="请输入"> </el-input>
- <el-row class="flex mt15">
- <el-button size="default" @click="onResetForm">
- 重置表单
- <el-button size="default" type="primary" @click="onSubmitForm">
- <SvgIcon name="iconfont icon-shuxing" />
- 验证表单
-import { formData } from './mock';
-// 定义接口来定义对象的类型
-interface FormDataOptions {
- label: string;
- value: string;
-interface FormDataState {
- prop: string;
- placeholder: string;
- clearable: boolean;
- disabled: boolean;
- required: boolean;
- type: string;
- i18n: boolean;
- i18nText: string;
- isShow: boolean;
- xs: number;
- sm: number;
- md: number;
- lg: number;
- xl: number;
- options?: FormDataOptions[];
-interface DynamicFormState {
- formData: FormDataState[];
- form: any;
- const { proxy } = <any>getCurrentInstance();
- const state = reactive<DynamicFormState>({
- formData,
- form: {
- name: '',
- email: '',
- autograph: '',
- occupation: '',
- list: [
- year: '',
- month: '',
- day: '',
- remarks: '',
- // 新增行
- const onAddRow = () => {
- state.form.list.push({
- // 删除行
- const onDelRow = (k: number) => {
- state.form.list.splice(k, 1);
- // 表单验证
- const onSubmitForm = () => {
- proxy.$refs.formRulesOneRef.validate((valid: boolean) => {
- proxy.$message.success('验证成功');
- return false;
- // 重置表单
- const onResetForm = () => {
- proxy.$refs.formRulesOneRef.resetFields();
- onAddRow,
- onDelRow,
- onSubmitForm,
- onResetForm,
@@ -1,119 +0,0 @@
-// 表单数据选项(自行扩展)
-export const formData = [
- label: '姓名',
- prop: 'name',
- placeholder: '请输入姓名',
- clearable: true,
- disabled: false,
- required: true,
- type: 'input',
- i18n: false,
- i18nText: '',
- isShow: true,
- xs: 24,
- sm: 12,
- md: 8,
- lg: 6,
- xl: 4,
- label: '邮箱',
- prop: 'email',
- placeholder: '请输入用户邮箱',
- label: '登陆时间',
- prop: 'autograph',
- placeholder: '选择时间',
- type: 'date',
- label: '职务',
- prop: 'occupation',
- placeholder: '请选择职务',
- type: 'select',
- options: [
- label: '计算机 / 互联网 / 通信',
- value: '1',
- label: '生产 / 工艺 / 制造',
- value: '2',
- label: '医疗 / 护理 / 制药',
- value: '3',
- label: '',
- prop: '',
- placeholder: '',
- type: '',
- sm: 24,
- md: 24,
- lg: 24,
- xl: 24,
- label: '备注',
- prop: 'remarks',
- placeholder: '请输入',
- type: 'textarea',
@@ -1,88 +0,0 @@
- <div class="element-container">
- <el-card shadow="hover" :header="`element plus 字体图标(自动载入,增加了 ele- 前缀,使用时:ele-Aim):${sheetsIconList.length}个`">
- <SvgIcon :name="v" :size="30" />
- // 初始化获取 css 样式,获取 element plus 自带 svg 图标,增加了 ele- 前缀,使用时:ele-Aim
- initIconfont.ele().then((res: any) => {
- state.sheetsIconList = res;
-.element-container {
@@ -1,34 +0,0 @@
- <div class="flex-margin color-primary">filtering-details 测试界面</div>
-import { computed, defineComponent } from 'vue';
- <div class="flex-margin color-primary">测试界面</div>
@@ -1,354 +0,0 @@
- <div class="filtering">
- <el-card
- shadow="hover"
- class="filtering-list br-top-no"
- v-loading="tableData.loading"
- element-loading-text="加载中..."
- element-loading-background="rgba(255, 255, 255, 0.1)"
- :class="{ 'min-h-360': tableData.data.length <= 0 }"
- <div
- v-for="(val, key) in filtering"
- :ref="
- (el) => {
- if (el) dlRefs[key] = el;
- "
- class="filtering-list-flex"
- <div class="filtering-list-title">{{ val.title }}</div>
- <div class="filtering-list-item" :style="{ height: val.isMore ? 'auto' : '50px' }">
- <span class="span" :class="v.active ? 'dd-active' : ''" v-for="(v, k) in val.children" :key="k" @click="onSelItem(val, v)">{{
- v.label
- }}</span>
- <div class="dd-more" v-if="val.isShowMore" @click="val.isMore = !val.isMore">
- <span>{{ val.isMore ? '收起' : '展开' }}</span>
- <i :class="val.isMore ? 'el-icon-arrow-down' : 'el-icon-arrow-right'"></i>
- <div class="flex-warp mt15 mb15" v-if="tableData.data.length > 0">
- <el-row :gutter="15">
- <el-col :xs="24" :sm="12" :md="8" :lg="6" :xl="4" class="mb15" v-for="(v, k) in tableData.data" :key="k" @click="onTableItemClick(v)">
- <div class="item-img">
- <img :src="v.img" />
- <div class="item-txt">
- <div class="item-txt-title">{{ v.title }}</div>
- <div class="item-txt-other">
- <div style="width: 100%">
- <div class="item-txt-msg mb10">
- <span>评价 {{ v.evaluate }}</span>
- <span class="ml10">收藏 {{ v.collection }}</span>
- <div class="item-txt-msg item-txt-price">
- <span class="font-price">
- <span>¥</span>
- <span class="font">{{ v.price }}</span>
- <span>月销{{ v.monSales }}笔</span>
- <div v-else class="filtering-no-data">
- <div class="no-data-box">
- <i class="el-icon-search"></i>
- <div class="no-txt">暂无数据</div>
- <template v-if="tableData.data.length > 0">
- <el-pagination
- style="text-align: right"
- background
- @size-change="onHandleSizeChange"
- @current-change="onHandleCurrentChange"
- :page-sizes="[10, 20, 30]"
- :current-page="tableData.param.pageNum"
- :page-size="tableData.param.pageSize"
- layout="total, sizes, prev, pager, next, jumper"
- :total="tableData.total"
- </el-pagination>
-import { ref, toRefs, reactive, onMounted, nextTick, defineComponent } from 'vue';
-import { useRouter } from 'vue-router';
-import { filtering, filterList } from './mock';
- const dlRefs: any = ref([]);
- const router = useRouter();
- filtering,
- tableData: {
- data: filterList,
- total: 99,
- loading: false,
- param: {
- pageNum: 1,
- pageSize: 10,
- initBtnToggle();
- window.onresize = () => {
- // 初始化 `收起、展开` 按钮
- const initBtnToggle = () => {
- const els = dlRefs.value;
- els.map((v: any, k: number) => {
- v.scrollHeight < v.lastChild.scrollHeight ? (state.filtering[k].isShowMore = true) : (state.filtering[k].isShowMore = false);
- // 过滤当前选中的数据
- const onSelItem = (val: any, v: any) => {
- val.children.map((v: any) => (v.active = false));
- v.active = true;
- let arr = [];
- state.filtering.map((item: any) => {
- item.children.map((chil: any) => {
- if (chil.active) {
- arr.push({
- ...item,
- children: [{ ...chil }],
- state.tableData.loading = true;
- setTimeout(() => {
- state.tableData.loading = false;
- }, 500);
- // 当前列表项点击
- const onTableItemClick = (v: any) => {
- if (v.id === 1) {
- router.push({
- path: '/pages/filtering/details',
- query: { id: v.id },
- path: '/pages/filtering/details1',
- // 分页点击
- const onHandleSizeChange = (val: number) => {
- state.tableData.param.pageSize = val;
- const onHandleCurrentChange = (val: number) => {
- state.tableData.param.pageNum = val;
- dlRefs,
- onSelItem,
- onTableItemClick,
- onHandleSizeChange,
- onHandleCurrentChange,
-.filtering {
- .filtering-list {
- border-bottom: none !important;
- .filtering-list-flex {
- &:last-of-type {
- .filtering-list-item {
- .filtering-list-title {
- float: left;
- width: 64px;
- font-weight: 700;
- color: #909399;
- margin: 15px 0;
- &:after {
- border: 1px solid #909399;
- border-width: 0 1px 1px 0;
- width: 4px;
- height: 4px;
- transform: rotate(-45deg) translateY(-50%);
- right: 10px;
- top: 50%;
- border-bottom: 1px dotted var(--next-border-color-light);
- margin-left: 64px;
- color: #8d8d91;
- .dd-active {
- .dd-more {
- right: 0;
- top: 16px;
- color: #a5a5a5;
- .br-top-no {
- border-top: none;
- .flex-warp {
- align-content: flex-start;
- margin: 0 -5px;
- height: 360px;
- border: 1px solid var(--next-border-color-light);
- box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.03);
- .item-txt-title {
- color: var(--el-color-primary) !important;
- .item-img {
- img {
- transform: translateZ(0) scale(1.05);
- height: 215px;
- .item-txt {
- -webkit-line-clamp: 2;
- -webkit-box-orient: vertical;
- display: -webkit-box;
- color: #666666;
- text-decoration: underline;
- .item-txt-other {
- align-items: flex-end;
- .item-txt-msg {
- .item-txt-price {
- .font-price {
- color: #ff5000;
- .font {
- font-size: 22px;
- ::v-deep(.el-card__body) {
- .filtering-no-data {
- .no-data-box {
- color: #cccccc;
- .no-txt {
- .min-h-360 {
@@ -1,201 +0,0 @@
-// 导航数据
-export const filtering = [
- title: '权限',
- isMore: false,
- isShowMore: false,
- id: 0,
- id: '01',
- label: '全部',
- active: true,
- id: '02',
- label: '普通用户',
- active: false,
- id: '03',
- label: '管理员',
- title: '布局',
- id: 1,
- id: 11,
- id: 12,
- label: '默认',
- id: 13,
- label: '经典',
- id: 14,
- label: '横向',
- id: 15,
- label: '分栏',
- title: '配置',
- id: 2,
- id: 21,
- id: 22,
- label: '开启 Breadcrumb',
- id: 23,
- label: '开启 Tags-View',
- id: 24,
- label: '固定 Header',
- id: 25,
- label: '侧边栏 Logo',
- id: 26,
- label: '开启折叠 NavMenu',
- id: 27,
- label: '开启一个 NavMenu 展开',
- id: 28,
- label: '登录用户头像',
-// 列表数据
-export const filterList = [
- img: 'http://news.sznews.com/pic/2020-08/14/9d9c9a60-f0af-41aa-b617-683b07c87642.jpg',
- title: '嘉陵江2020年第1号洪水”在嘉陵江支流涪江形成',
- evaluate: (Math.random() * 10).toFixed(2),
- collection: (Math.random() * 100).toFixed(2),
- price: (Math.random() * 10).toFixed(2),
- monSales: (Math.random() * 20).toFixed(2),
- img: 'http://www.sznews.com/news/pic/2020-08/13/0ea47d3c-feb9-4bd7-8597-a8a373aa6340c6ec12c7-3b33-4528-91a6-85ec8ca1df67_watermark.png',
- title: '让《民法典》走近群众 盐田街道开展人民调解宣传活动',
- img: 'http://www.sznews.com/photo/pic/2020-08/12/a08d6eb0-1d53-4f76-a313-ad3e5d701f98.jpg',
- title: '记者手记:可可西里,“挪”向“藏羚羊大产房”的14个半小时',
- id: 3,
- img: 'http://www.sznews.com/photo/pic/2020-08/11/43cc0e14-9bca-45b9-9a8b-342e09d6a4c7.jpg',
- title: '以优异成绩庆祝深圳经济特区建立40周年',
- id: 4,
- img: 'http://www.sznews.com/photo/pic/2020-08/11/a4dc322b-68ec-40e6-8906-3124142c3e49.jpg',
- title: '草原上的“太阳姑娘”',
- id: 5,
- img: 'http://www.sznews.com/zhuanti/pic/2020-08/07/57f087b4-4812-46cc-adb9-ead73621284e.png',
- title: '奇观天下|带你走进非洲野生动物观光第一目的地',
- id: 6,
- img: 'http://news.sznews.com/pic/2020-09/02/t2_(101X54X600X335)7cd39301-d9cf-45f1-91c3-9575b1e5ce0e.jpg.2',
- title: '五角大楼发布“中国军力报告” 华春莹: 罔顾事实,充满偏见',
- id: 7,
- img: 'http://news.sznews.com/pic/2020-09/02/b8b41d9c-0508-4498-8d37-6e597493769f.jpg',
- title: '最新地铁消息汇总:4号线北延、2号线三期、8号线一期等今年通车',
- id: 8,
- img: 'http://www.sznews.com/photo/pic/2020-08/10/1635374c-f4d6-475c-ac47-1334176f365d.png',
- title: '9月1日深圳新增5例无症状感染者!钟南山这段话冲上热搜!',
- id: 9,
- img: 'http://www.sznews.com/news/pic/2020-08/13/646e5458-92b7-4636-9940-9b0799babfe1.png',
- title: '全能“小福宝” 为文明社区建设添砖加瓦',
- id: 10,
@@ -1,113 +0,0 @@
- <div class="form-adapt-container">
- <el-card shadow="hover" header="表单自适应演示(改变窗口查看效果)">
- <el-form :model="form" size="default" label-width="100px" class="mt35 mb35">
- <el-col :xs="24" :sm="12" :md="8" :lg="6" :xl="4" class="mb20">
- <el-form-item label="姓名">
- <el-input v-model="form.name" placeholder="请输入姓名" clearable></el-input>
- <el-form-item label="用户归属部门">
- <el-input v-model="form.email" placeholder="请输入用户归属部门" clearable></el-input>
- <el-form-item label="登陆账户名">
- <el-input v-model="form.autograph" placeholder="请输入登陆账户名" clearable></el-input>
- <el-form-item label="职务">
- <el-select v-model="form.occupation" placeholder="请选择职务" clearable class="w100">
- <el-option label="计算机 / 互联网 / 通信" value="1"></el-option>
- <el-option label="生产 / 工艺 / 制造" value="2"></el-option>
- <el-option label="医疗 / 护理 / 制药" value="3"></el-option>
- <el-form-item label="手机">
- <el-input v-model="form.phone" placeholder="请输入手机" clearable></el-input>
- <el-form-item label="性别">
- <el-select v-model="form.sex" placeholder="请选择性别" clearable class="w100">
- <el-option label="男" value="1"></el-option>
- <el-option label="女" value="2"></el-option>
- <el-form-item label="登录密码">
- <el-input v-model="form.phone1" placeholder="请输入登录密码" clearable></el-input>
- <el-form-item label="权限角色">
- <el-input v-model="form.phone2" placeholder="请输入权限角色" clearable></el-input>
- <el-form-item label="创建用户">
- <el-input v-model="form.phone3" placeholder="请输入创建用户" clearable></el-input>
- <el-form-item label="修改用户">
- <el-input v-model="form.phone4" placeholder="请输入修改用户" clearable></el-input>
- <el-form-item label="所属用户">
- <el-input v-model="form.phone5" placeholder="请输入所属用户" clearable></el-input>
- <el-form-item label="所属部门">
- <el-input v-model="form.phone6" placeholder="请输入所属部门" clearable></el-input>
- <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
- <el-form-item>
- <el-button type="primary">
- <SvgIcon name="iconfont icon-biaodan" />
- 更新个人信息
- phone: '',
- sex: '',
- phone1: '',
- phone2: '',
- phone3: '',
- phone4: '',
- phone5: '',
- phone6: '',
@@ -1,58 +0,0 @@
- <div class="form-i18n-container">
- <el-card shadow="hover" header="表单国际化演示(不适用于动态项 form-item)">
- <div style="text-align: center; margin-top: 15px">
- <el-radio-group v-model="radio" size="default" @change="onRadioChange">
- <el-radio-button label="zh-cn">中文简体</el-radio-button>
- <el-radio-button label="en">英文</el-radio-button>
- <el-radio-button label="zh-tw">中文繁体</el-radio-button>
- </el-radio-group>
- <el-col :xs="24" :sm="12" :md="8" :lg="8" :xl="8" class="mb20">
- <el-form-item :label="$t('message.formI18nLabel.name')">
- <el-input v-model="form.name" :placeholder="$t('message.formI18nPlaceholder.name')" clearable></el-input>
- <el-form-item :label="$t('message.formI18nLabel.email')">
- <el-input v-model="form.email" :placeholder="$t('message.formI18nPlaceholder.email')" clearable></el-input>
- <el-form-item :label="$t('message.formI18nLabel.autograph')">
- <el-input v-model="form.autograph" :placeholder="$t('message.formI18nPlaceholder.autograph')" clearable></el-input>
-import { toRefs, reactive, defineComponent, getCurrentInstance } from 'vue';
- radio: 'zh-cn',
- // 单选框改变时
- const onRadioChange = () => {
- proxy.$i18n.locale = state.radio;
- onRadioChange,
@@ -1,67 +0,0 @@
- <div class="form-rules-one-container">
- <el-form :model="form" :rules="rules" ref="formRulesOneRef" size="default" label-width="100px" class="mt35">
- <el-form-item label="姓名" prop="name">
- <el-form-item label="邮箱" prop="email">
- <el-input v-model="form.email" placeholder="请输入用户邮箱" clearable></el-input>
- <el-form-item label="登陆账户名" prop="autograph">
- <el-form-item label="职务" prop="occupation">
- name: 'pagesFormRulesOne',
- props: {
- data: {
- type: Object,
- default: () => {},
- setup(props) {
- form: { name: '', email: '', autograph: '', occupation: '' },
- rules: {
- name: { required: true, message: '请输入姓名', trigger: 'blur' },
- email: { required: true, message: '请输入用户邮箱', trigger: 'blur' },
- autograph: { required: true, message: '请输入登陆账户名', trigger: 'blur' },
- occupation: { required: true, message: '请选择职务', trigger: 'change' },
- // 赋值回显
- const initForm = () => {
- state.form = <any>props.data;
- initForm();
@@ -1,49 +0,0 @@
- <div class="form-rules-three-container">
- <el-form :model="form" :rules="rules" ref="formRulesThreeRef" size="default" label-width="100px" class="mt35">
- <el-form-item label="创建用户" prop="createUser">
- <el-input v-model="form.createUser" placeholder="请输入创建用户" clearable></el-input>
- <el-form-item label="修改用户" prop="editUser">
- <el-input v-model="form.editUser" placeholder="请输入修改用户" clearable></el-input>
- <el-form-item label="所属用户" prop="user">
- <el-input v-model="form.user" placeholder="请输入所属用户" clearable></el-input>
- <el-form-item label="所属部门" prop="department">
- <el-input v-model="form.department" placeholder="请输入所属部门" clearable></el-input>
- name: 'pagesFormRulesThree',
- form: { createUser: '', editUser: '', user: '', department: '' },
- createUser: { required: true, message: '请输入创建用户', trigger: 'blur' },
- editUser: { required: true, message: '请输入修改用户', trigger: 'blur' },
- user: { required: true, message: '请输入所属用户', trigger: 'blur' },
- department: { required: true, message: '请输入所属部门', trigger: 'blur' },
@@ -1,51 +0,0 @@
- <div class="form-rules-two-container">
- <el-form :model="form" :rules="rules" ref="formRulesTwoRef" size="default" label-width="100px" class="mt35">
- <el-form-item label="手机" prop="phone">
- <el-form-item label="登录密码" prop="password">
- <el-input v-model="form.password" placeholder="请输入登录密码" clearable></el-input>
- <el-form-item label="权限角色" prop="auth">
- <el-input v-model="form.auth" placeholder="请输入权限角色" clearable></el-input>
- name: 'pagesFormRulesTwo',
- form: { phone: '', sex: '', password: '', auth: '' },
- phone: { required: true, message: '请输入手机', trigger: 'blur' },
- password: { required: true, message: '请输入登录密码', trigger: 'blur' },
- auth: { required: true, message: '请输入权限角色', trigger: 'blur' },
@@ -1,85 +0,0 @@
- <div class="form-rules-container">
- <el-card shadow="hover" header="表单组件1">
- <FormRulesOne :data="formRulesOneData" ref="pagesFormRulesOneRef" />
- <el-card shadow="hover" header="表单组件2" class="mt15">
- <FormRulesTwo ref="pagesFormRulesTwoRef" />
- <el-card shadow="hover" header="表单组件3" class="mt15">
- <FormRulesThree ref="pagesFormRulesThreeRef" />
- <SvgIcon name="ele-RefreshRight" />
-import { ElMessage } from 'element-plus';
-import FormRulesOne from './component/formRulesOne.vue';
-import FormRulesTwo from './component/formRulesTwo.vue';
-import FormRulesThree from './component/formRulesThree.vue';
- components: {
- FormRulesOne,
- FormRulesTwo,
- FormRulesThree,
- formRulesOneData: {
- name: 'lyt',
- email: 'lyt123@.com',
- autograph: 'lyt123456',
- occupation: '1',
- // 表单组件验证
- const formRulesValidate = (pageRef: string, sonRef: string) => {
- return new Promise((resolve) => {
- proxy.$refs[pageRef].$refs[sonRef].validate((valid: boolean) => {
- if (valid) resolve(valid);
- // 表单组件重置
- const formRulesResetFields = () => {
- proxy.$refs.pagesFormRulesOneRef.$refs.formRulesOneRef.resetFields();
- proxy.$refs.pagesFormRulesTwoRef.$refs.formRulesTwoRef.resetFields();
- proxy.$refs.pagesFormRulesThreeRef.$refs.formRulesThreeRef.resetFields();
- // 验证表单
- Promise.all([
- formRulesValidate('pagesFormRulesOneRef', 'formRulesOneRef'),
- formRulesValidate('pagesFormRulesTwoRef', 'formRulesTwoRef'),
- formRulesValidate('pagesFormRulesThreeRef', 'formRulesThreeRef'),
- ]).then(() => {
- ElMessage.success('表单全部验证成功');
- formRulesResetFields();
- <div class="iconfont-container">
- <el-card shadow="hover" :header="`iconfont 字体图标(自动载入):${sheetsIconList.length}个`">
- <i :class="v" class="iconfont"></i>
- // 初始化获取 css 样式,这里使用阿里的图标(记得加上前缀 `iconfont`),其它第三方请自行做判断
- initIconfont.ali().then((res: any) => (state.sheetsIconList = res));
-.iconfont-container {
@@ -1,193 +0,0 @@
- <div class="lazy-img-container">
- <el-card shadow="hover" header="图片懒加载演示(F12 切换到 Network Img下进行图片加载查看)">
- <div class="flex-warp" v-if="tableData.data.length > 0">
- <div class="item-img" v-loading="v.loading">
- <img :data-img="v.img" :data-key="k" :data-lazy-img-list="k" />
- <el-empty v-else description="暂无数据"></el-empty>
-import other from '/@/utils/other';
-import { filterList } from './mock';
- path: '/pages/filteringDetails',
- other.lazyImg('[data-lazy-img-list]', state.tableData.data);
-.lazy-img-container {
@@ -1,313 +0,0 @@
- img: 'https://news.sznews.com/pic/2021-03/09/e37326cc-4583-48f3-aa00-ecc2392d319d.jpg',
- title: '36分钟,深圳平均通勤时间出炉!GDP10强城市中仅输杭州',
- loading: true,
- img: 'http://news.sznews.com/pic/2021-03/09/78cf72b6-e2d9-459d-a368-470414a027f4679cf4ea-26fa-48c8-9fee-c2d092a91400.png',
- title: '为爱而动,“红色鹊桥”三八妇女节交友联谊活动助力深圳女孩脱单',
- img: 'http://news.sznews.com/pic/2021-03/09/1faf3c6e-1250-4e6b-b072-4a331553e027.jpg',
- title: '粤桂协作“背水一战” 解决广西大化县3.7万人饮水难题',
- img: 'https://news.sznews.com/pic/2021-03/09/9fcf6dd4-1e80-4497-bdc9-83dc7246d170.jpg.2',
- title: '城镇就业女性平均薪酬6847元 女性职场渗透率提升',
- img: 'https://news.sznews.com/pic/2021-03/09/1bd78227-4126-4a43-bdf6-48ead6edd1bf.jpg.2',
- title: '深圳:实现“从0到1”源头创新,推进大湾区综合性国家科学中心建设!',
- img: 'http://news.sznews.com/pic/2021-03/08/9ea943a3-3ae8-4f49-8296-711ec36ef8c6_watermark.png',
- title: '煖声音第126期|愿你有诗酒趁年华的洒脱,也有岁月沉淀后的坚定从容',
- img: 'https://news.sznews.com/pic/2021-03/08/a95ba232-1422-4f7e-b85f-c61d486c8659.jpg.2',
- title: '姐妹们一起来吐槽,最不能接受男人的缺点!',
- img: 'http://news.sznews.com/pic/2021-03/08/76816bf0-3899-4c7e-bc6e-079b5ba8725e.jpg',
- title: '民生小事 | 手机遗落出租车 热心民警帮找回',
- img: 'https://news.sznews.com/pic/2021-03/08/28ed70d4-71f5-4abb-bf7b-0294bece9e43.jpg.2',
- title: '“十三五”:深圳交上靓丽答卷 发展动力加快转换',
- img: 'http://news.sznews.com/pic/2021-03/05/d13ae31f-fd45-431a-b48e-c5895bbc193e.png',
- title: '深圳湾公园一女子落水,三名男子接力及时施救',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210704/653/w930h523/20210704/d5d2-krwipas6444058.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210704/766/w930h636/20210704/b1ae-krwipas6332914.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210704/750/w930h620/20210704/2886-krwipas6264821.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210704/750/w930h620/20210704/767c-krwipas6387862.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210704/111/w1024h687/20210704/1f65-krwipas5871436.jpg',
- title: '盛夏的那考河湿地公园!',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210704/657/w930h527/20210704/7eae-krwipas5866609.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210703/760/w930h630/20210703/124e-krwipas5596390.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210703/27/w930h697/20210703/9630-krwipas5514972.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210703/750/w930h620/20210703/2fe3-krwipas5388050.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210703/724/w930h594/20210703/98b6-krwipas5234060.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210703/750/w930h620/20210703/f765-krwipas5194727.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210702/750/w930h620/20210702/5dde-krwipas4724976.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210702/750/w930h620/20210702/f45e-krwipas4566804.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210702/750/w930h620/20210702/5579-krwipas4551382.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210702/750/w930h620/20210702/7c75-krwipas4543661.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210702/653/w930h523/20210702/ece2-krwipas4411140.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210702/750/w930h620/20210702/f5c2-krwipas4215211.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210701/720/w930h590/20210701/eabc-krwipas3509204.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210701/797/w930h667/20210701/4667-krwipas3365057.jpg',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210701/750/w930h620/20210701/baea-krwipas2976622.jpg',
- title: '民众前往中共一大纪念馆参观',
- img: 'http://z0.sinaimg.cn/auto/resize?size=235_156&img=http://n.sinaimg.cn/spider20210630/617/w850h567/20210630/5c96-krwipas1819108.jpg',
- title: '延吉灯光秀美轮美奂 市民徜徉璀璨夜景',
@@ -1,209 +0,0 @@
- <div class="list-adapt-container">
- <el-card shadow="hover" header="列表自适应演示(改变窗口查看效果)">
-interface ListAdaptRow {
- img: string;
- title: string;
- evaluate: string;
- collection: string;
- price: string;
- monSales: string;
- id: number;
-interface TableDataState {
- data: Array<ListAdaptRow>;
- total: number;
- loading: boolean;
- pageNum: number;
- pageSize: number;
- const state = reactive<TableDataState>({
- const onTableItemClick = (v: ListAdaptRow) => {
-.flex-warp {
@@ -1,93 +0,0 @@
@@ -1,27 +0,0 @@
- <div class="preview-container">
- <el-card shadow="hover" header="element-plus 大图预览">
- <el-image style="width: 100px; height: 100px; border-radius: 5px" :src="url" :preview-src-list="srcList" title="点击查看大图预览"> </el-image>
- url: 'https://ss0.bdstatic.com/70cFvHSh_Q1YnxGkpoWK1HF6hhy/it/u=1813762643,1914315241&fm=26&gp=0.jpg',
- srcList: [
- 'https://ss0.bdstatic.com/70cFvHSh_Q1YnxGkpoWK1HF6hhy/it/u=1813762643,1914315241&fm=26&gp=0.jpg',
- 'https://ss1.bdstatic.com/70cFuXSh_Q1YnxGkpoWK1HF6hhy/it/u=317673774,2961727727&fm=26&gp=0.jpg',
- 'https://fuss10.elemecdn.com/1/8e/aeffeb4de74e2fde4bd74fc7b4486jpeg.jpeg',
@@ -1,50 +0,0 @@
- <div class="steps-container">
- <el-card shadow="hover" header="element-plus 步骤条">
- <el-steps :active="stepsActive">
- <el-step title="第一步">
- <template #icon>
- <SvgIcon name="iconfont icon-0_round_solid" :size="20" />
- </el-step>
- <el-step title="第二步">
- <SvgIcon name="iconfont icon-2_round_solid" :size="20" />
- <el-step title="第三步">
- <SvgIcon name="iconfont icon-3_round_solid" :size="20" />
- </el-steps>
- <el-result icon="success" title="成功提示" subTitle="请根据提示进行操作" v-if="stepsActive === 1"> </el-result>
- <el-result icon="warning" title="警告提示" subTitle="请根据提示进行操作" v-else-if="stepsActive === 2"> </el-result>
- <el-result icon="error" title="错误提示" subTitle="请根据提示进行操作" v-else-if="stepsActive === 3"> </el-result>
- <el-button @click="onNextSteps" size="default" class="mt15" type="primary">
- <SvgIcon name="iconfont icon-step" />
- 下一步
- stepsActive: 1,
- // 下一步点击
- const onNextSteps = () => {
- if (state.stepsActive++ > 2) state.stepsActive = 1;
- onNextSteps,
@@ -1,129 +0,0 @@
- <el-card shadow="hover" header="表单表格验证">
- <el-form ref="tableRulesRef" :model="tableData" size="default">
- <el-table :data="tableData.data" border class="module-table-uncollected">
- <el-table-column
- v-for="(item, index) in tableData.header"
- :key="index"
- show-overflow-tooltip
- :prop="item.prop"
- :width="item.width"
- :label="item.label"
- <template v-slot:header>
- <span v-if="item.isRequired" class="color-danger">*</span>
- <span class="pl5">{{ item.label }}</span>
- <el-tooltip v-if="item.isTooltip" effect="dark" content="这是tooltip" placement="top">
- <i class="iconfont icon-quanxian" />
- </el-tooltip>
- <template v-slot="scope">
- :prop="`data.${scope.$index}.${item.prop}`"
- :rules="[{ required: item.isRequired, message: '不能为空', trigger: `${item.type}` == 'input' ? 'blur' : 'change' }]"
- <el-select v-if="item.type === 'select'" v-model="scope.row[item.prop]" placeholder="请选择">
- <el-option v-for="sel in tableData.option" :key="sel.id" :label="sel.label" :value="sel.value" />
- v-else-if="item.type === 'date'"
- v-model="scope.row[item.prop]"
- placeholder="选择日期"
- <el-input v-else-if="item.type === 'input'" v-model="scope.row[item.prop]" placeholder="请输入内容" />
- <el-input v-else-if="item.type === 'dialog'" v-model="scope.row[item.prop]" readonly placeholder="请输入内容">
- <template v-slot:suffix>
- <i class="iconfont icon-shouye_dongtaihui" />
- </el-table-column>
- <el-button size="default" type="success" @click="onValidate">表格验证</el-button>
- <el-button size="default" type="primary" @click="onAddRow">新增一行</el-button>
-import { defineComponent, toRefs, reactive, ref } from 'vue';
-interface TableHeader {
- width: string | number;
- isRequired?: boolean;
- isTooltip?: boolean;
-interface TableRulesState {
- data: any[];
- header: TableHeader[];
- option: any[];
- const tableRulesRef = ref();
- const state = reactive<TableRulesState>({
- data: [],
- header: [
- { prop: 'a1', width: '', label: '一级分类', isRequired: true, type: 'select' },
- { prop: 'a2', width: '', label: '二级分类', isRequired: true, type: 'select' },
- { prop: 'a3', width: '', label: '三级分类', isRequired: true, type: 'select' },
- { prop: 'a4', width: '', label: '四级分类', isRequired: true, type: 'date' },
- { prop: 'a5', width: '', label: '五级分类', isRequired: true, type: 'input' },
- { prop: 'a6', width: '', label: '六级分类', isTooltip: true, type: 'dialog' },
- { prop: 'a7', width: '', label: '演示级分类', type: 'input' },
- { prop: 'a8', width: '', label: '颜色是分类', type: 'input' },
- option: [
- { value: '选项1', label: '黄金糕' },
- { value: '选项2', label: '双皮奶' },
- { value: '选项3', label: '蚵仔煎' },
- // 表格验证
- const onValidate = () => {
- if (state.tableData.data.length <= 0) return ElMessage.warning('请先点击增加一行');
- tableRulesRef.value.validate((valid: any) => {
- if (!valid) return ElMessage.warning('表格项必填未填');
- ElMessage.success('全部验证通过');
- // 新增一行
- state.tableData.data.push({
- a1: '',
- a2: '',
- a3: '',
- a6: '',
- a7: '',
- a8: '',
- onValidate,
- tableRulesRef,
@@ -1,258 +0,0 @@
- <div class="tree-container">
- <el-card shadow="hover" header="element plus Tree 树形控件改成表格">
- <div v-loading="treeLoading">
- <div class="tree-head">
- <div class="tree-head-check"><el-checkbox v-model="treeCheckAll" @change="onCheckAllChange"></el-checkbox></div>
- <div class="tree-head-one">商品 ID</div>
- <div style="flex: 1; display: flex">
- <div class="tree-head-two">商品名称</div>
- <div class="tree-head-three">描述</div>
- <el-tree :data="treeTableData" show-checkbox node-key="id" ref="treeTable" :props="treeDefaultProps" @check="onCheckTree">
- <template #default="{ node, data }">
- <span class="tree-custom-node">
- <span style="flex: 1">{{ node.label }}</span>
- <span v-if="data.isShow" style="flex: 1; display: flex">
- <span type="text" size="default" style="flex: 1">{{ data.label1 }}</span>
- <span type="text" size="default" style="flex: 1">{{ data.label2 }}</span>
- </el-tree>
- <el-button @click="onSelect" class="mt15" size="default" type="primary">
- <SvgIcon name="iconfont icon-shuxingtu" />
- 选择元素
-import { toRefs, reactive, onBeforeMount, getCurrentInstance, defineComponent } from 'vue';
-interface TreeDataState {
- label1: string;
- label2: string;
- children?: TreeDataState[];
-interface TreeSate {
- treeCheckAll: boolean;
- treeLoading: boolean;
- treeTableData: TreeDataState[];
- treeDefaultProps: {
- children: string;
- treeSelArr: TreeDataState[];
- treeLength: number;
- const state = reactive<TreeSate>({
- treeCheckAll: false,
- treeLoading: false,
- treeTableData: [],
- children: 'children',
- label: 'label',
- treeSelArr: [],
- treeLength: 0,
- // 初始化树的长度
- const initTreeLengh = (arr: TreeDataState[]) => {
- let count = 0;
- arr.map((item) => {
- if (item.children) {
- count += item.children.length;
- state.treeLength = count + arr.length;
- // 全选改变时
- const onCheckAllChange = () => {
- if (state.treeCheckAll) {
- proxy.$refs.treeTable.setCheckedNodes(state.treeTableData);
- proxy.$refs.treeTable.setCheckedKeys([]);
- // 节点选中状态发生变化时的回调
- const onCheckTree = () => {
- state.treeSelArr = [];
- state.treeSelArr = proxy.$refs.treeTable.getCheckedNodes();
- state.treeSelArr.length == state.treeLength ? (state.treeCheckAll = true) : (state.treeCheckAll = false);
- // 选择元素按钮
- const onSelect = () => {
- let treeArr = proxy.$refs.treeTable.getCheckedNodes();
- if (treeArr.length <= 0) {
- ElMessage.warning('请选择元素');
- return;
- // console.log(proxy.$refs.treeTable.getCheckedNodes());
- // 初始化树模拟数据
- const getTreeData = () => {
- state.treeTableData = [
- label: '12987121',
- label1: '好滋好味鸡蛋仔',
- label2: '荷兰优质淡奶,奶香浓而不腻',
- label: '一级 1-1',
- isShow: false,
- label: '一级 1-2',
- label: '12987122',
- label: '二级 2-1',
- label: '二级 2-2',
- label: '12987123',
- id: 31,
- label: '二级 3-1',
- id: 32,
- label: '二级 3-2',
- id: 33,
- label: '二级 3-3',
- ];
- initTreeLengh(state.treeTableData);
- getTreeData();
- getTreeData,
- onCheckAllChange,
- onCheckTree,
- onSelect,
-.tree-container {
- .tree-head {
- height: 48px;
- line-height: 48px;
- border-bottom: none;
- padding-right: 8px;
- .tree-head-check {
- width: 38px;
- .tree-head-one,
- .tree-head-two,
- .tree-head-three {
- .tree-head-one {
- padding-left: 8px;
- .el-tree {
- .tree-custom-node {
- &::v-deep(.el-tree-node) {
- .el-tree-node__content {
- line-height: 57px !important;
- height: 57px !important;
- .el-tree-node__children {
- .el-tree-node {
- border: none;
@@ -1,173 +0,0 @@
- <div class="waterfall-container">
- <el-card shadow="hover" header="瀑布屏(布局一)" class="mb15">
- <div class="waterfall-first">
- <div class="waterfall-first-item" v-for="v in 30" :key="v" v-waves>
- <span class="flex-margin">{{ v }}</span>
- <el-card shadow="hover" header="瀑布屏(布局二)">
- <div class="waterfall-last">
- <div class="waterfall-last-item" v-for="v in 30" :key="v" v-waves="'light'">
-.waterfall-container {
- .waterfall-first {
- display: grid;
- grid-template-columns: repeat(auto-fill, minmax(188px, 1fr));
- grid-gap: 0.25em;
- grid-auto-flow: row dense;
- grid-auto-rows: 20px;
- .waterfall-first-item {
- border-radius: 3px;
- &:nth-of-type(3n + 1) {
- grid-row: auto / span 5;
- &:nth-of-type(3n + 2) {
- grid-row: auto / span 6;
- &:nth-of-type(3n + 3) {
- grid-row: auto / span 8;
- .waterfall-last {
- grid-auto-rows: minmax(188px, 20vmin);
- grid-template-columns: 1fr;
- .waterfall-last-item {
- @media (min-width: 576px) {
- grid-template-columns: repeat(7, 1fr);
- &:nth-of-type(9n + 9) {
- grid-column: auto / span 2;
- &:nth-of-type(9n + 8) {
- &:nth-of-type(9n + 7) {
- grid-column: auto / span 3;
- &:nth-of-type(9n + 6) {
- &:nth-of-type(9n + 5) {
- &:nth-of-type(9n + 4) {
- &:nth-of-type(9n + 3) {
- &:nth-of-type(9n + 2) {
- &:nth-of-type(9n + 1) {
- @media (min-width: 576px) and (min-width: 1024px) {
- grid-template-columns: repeat(14, 1fr);
- &:nth-of-type(15n + 15) {
- &:nth-of-type(15n + 14) {
- &:nth-of-type(15n + 13) {
- &:nth-of-type(15n + 12) {
- &:nth-of-type(15n + 11) {
- &:nth-of-type(15n + 10) {
- &:nth-of-type(15n + 9) {
- &:nth-of-type(15n + 8) {
- &:nth-of-type(15n + 7) {
- &:nth-of-type(15n + 6) {
- &:nth-of-type(15n + 5) {
- &:nth-of-type(15n + 4) {
- &:nth-of-type(15n + 3) {
- &:nth-of-type(15n + 2) {
- &:nth-of-type(15n + 1) {
@@ -1,133 +0,0 @@
- <el-card shadow="hover" header="波浪指令效果(v-waves)作用于 btn">
- <el-row class="mb10" style="color: #808080">可选参数 v-waves=" |light|red|orange|purple|green|teal"</el-row>
- <el-button size="default" v-waves>
- <SvgIcon name="iconfont icon-bolangnengshiyanchang" />
- 默认效果
- <el-button type="primary" size="default" v-waves="'light'">
- light 效果
- <el-button type="success" size="default" v-waves="'red'">
- red 效果
- <el-button type="info" size="default" v-waves="'orange'">
- orange 效果
- <el-button type="warning" size="default" v-waves="'purple'">
- purple 效果
- <el-button type="danger" size="default" v-waves="'green'">
- green 效果
- <el-button type="primary" size="default" v-waves="'teal'">
- teal 效果
- <el-card shadow="hover" header="波浪指令效果(v-waves)作用于 div" class="mt15">
- <div class="waterfall-first-item" v-for="v in 12" :key="v" v-waves>
-.preview-container {
- <transition name="el-zoom-in-center">
- aria-hidden="true"
- class="el-dropdown__popper el-popper is-light is-pure custom-contextmenu"
- role="tooltip"
- data-popper-placement="bottom"
- :style="`top: ${dropdowns.y + 5}px;left: ${dropdowns.x}px;`"
- :key="Math.random()"
- v-show="isShow"
- <ul class="el-dropdown-menu">
- <li
- v-for="(v, k) in dropdownList"
- class="el-dropdown-menu__item"
- aria-disabled="false"
- tabindex="-1"
- :key="k"
- @click="onCurrentClick(v.contextMenuClickId)"
- <SvgIcon :name="v.icon" />
- <span>{{ v.txt }}{{ item.type === 'line' ? '线' : '节点' }}</span>
- </li>
- <div class="el-popper__arrow" style="left: 10px"></div>
- </transition>
-import { computed, defineComponent, reactive, toRefs, onMounted, onUnmounted } from 'vue';
- name: 'pagesWorkflowContextmenu',
- dropdown: {
- setup(props, { emit }) {
- dropdownList: [
- { contextMenuClickId: 0, txt: '删除', icon: 'ele-Delete' },
- { contextMenuClickId: 1, txt: '编辑', icon: 'ele-Edit' },
- item: {
- type: 'node',
- conn: {},
- // 父级传过来的坐标 x,y 值
- const dropdowns = computed(() => {
- return <any>props.dropdown;
- // 当前项菜单点击
- const onCurrentClick = (contextMenuClickId: number) => {
- emit('current', Object.assign({}, { contextMenuClickId }, state.item), state.conn);
- // 打开右键菜单:判断是否固定,固定则不显示关闭按钮
- const openContextmenu = (item: any, conn = {}) => {
- state.item = item;
- state.conn = conn;
- closeContextmenu();
- state.isShow = true;
- }, 10);
- // 关闭右键菜单
- const closeContextmenu = () => {
- state.isShow = false;
- // 监听页面监听进行右键菜单的关闭
- document.body.addEventListener('click', closeContextmenu);
- document.body.addEventListener('contextmenu', closeContextmenu);
- // 页面卸载时,移除右键菜单监听事件
- document.body.removeEventListener('click', closeContextmenu);
- document.body.removeEventListener('contextmenu', closeContextmenu);
- dropdowns,
- openContextmenu,
- closeContextmenu,
- onCurrentClick,
-.custom-contextmenu {
- transform-origin: center top;
- z-index: 2190;
- position: fixed;
- .el-dropdown-menu__item {
@@ -1,73 +0,0 @@
- <el-drawer :title="`${nodeData.type === 'line' ? '线' : '节点'}操作`" v-model="isOpen" size="320px">
- <el-scrollbar>
- <Line v-if="nodeData.type === 'line'" @change="onLineChange" @close="close" ref="lineRef" />
- <Node v-else @submit="onNodeSubmit" @close="close" ref="nodeRef" />
- </el-scrollbar>
- </el-drawer>
-import { defineComponent, reactive, toRefs, ref, nextTick } from 'vue';
-import Line from './line.vue';
-import Node from './node.vue';
-interface WorkflowDrawerState {
- isOpen: boolean;
- nodeData: {
- jsplumbConn: any;
- name: 'pagesWorkflowDrawer',
- components: { Line, Node },
- const lineRef = ref();
- const nodeRef = ref();
- const state = reactive<WorkflowDrawerState>({
- isOpen: false,
- jsplumbConn: {},
- // 打开抽屉
- const open = (item: any, conn: any) => {
- state.isOpen = true;
- state.jsplumbConn = conn;
- state.nodeData = item;
- if (item.type === 'line') lineRef.value.getParentData(item);
- else nodeRef.value.getParentData(item);
- // 关闭
- const close = () => {
- state.isOpen = false;
- // 线 label 内容改变时
- const onLineChange = (label: any) => {
- state.jsplumbConn.label = label;
- emit('label', state.jsplumbConn);
- // 节点内容改变时
- const onNodeSubmit = (data: object) => {
- emit('node', data);
- lineRef,
- nodeRef,
- open,
- close,
- onLineChange,
- onNodeSubmit,
- <div class="pt15 pr15 pb15 pl15">
- <el-form :model="line" size="default" label-width="50px">
- <el-form-item label="来往">
- <el-input v-model="line.contact" placeholder="来往" clearable disabled></el-input>
- <el-form-item label="类型">
- <el-input v-model="line.type" placeholder="类型" clearable disabled></el-input>
- <el-form-item label="label">
- <el-input v-model="line.label" placeholder="请输入label内容" clearable></el-input>
- <el-button @click="onLineTextReset">
- 重置
- <el-button @click="onLineTextChange" type="primary">
- <SvgIcon name="ele-Check" />
- 保存
-import { defineComponent, reactive, toRefs } from 'vue';
-interface WorkflowDrawerLineState {
- line: any;
- name: 'pagesWorkflowDrawerLine',
- const state = reactive<WorkflowDrawerLineState>({
- line: {},
- // 获取父组件数据
- const getParentData = (data: object) => {
- state.line = data;
- // 重置
- const onLineTextReset = () => {
- state.line.label = '';
- // 保存
- const onLineTextChange = () => {
- emit('change', state.line.label);
- emit('close');
- getParentData,
- onLineTextReset,
- onLineTextChange,
@@ -1,272 +0,0 @@
- <div class="workflow-drawer-node">
- <el-tabs type="border-card" v-model="tabsActive">
- <!-- 节点编辑 -->
- <el-tab-pane label="节点编辑" name="1">
- <el-form :model="node" :rules="nodeRules" ref="nodeFormRef" size="default" label-width="80px" class="pt15 pr15 pb15 pl15">
- <el-form-item label="数据id" prop="id">
- <el-input v-model="node.id" placeholder="请输入数据id" clearable disabled></el-input>
- <el-form-item label="节点id" prop="nodeId">
- <el-input v-model="node.nodeId" placeholder="请输入节点id" clearable disabled></el-input>
- <el-form-item label="类型" prop="type">
- <el-input v-model="node.type" placeholder="请输入类型" clearable disabled></el-input>
- <el-form-item label="left坐标" prop="left">
- <el-input v-model="node.left" placeholder="请输入left坐标" clearable disabled></el-input>
- <el-form-item label="top坐标" prop="top">
- <el-input v-model="node.top" placeholder="请输入top坐标" clearable disabled></el-input>
- <el-form-item label="icon图标" prop="icon">
- <el-input v-model="node.icon" placeholder="请输入icon图标" clearable></el-input>
- <el-form-item label="名称" prop="name">
- <el-input v-model="node.name" placeholder="请输入名称" clearable></el-input>
- <el-button class="mb15" @click="onNodeRefresh">
- <el-button type="primary" class="mb15" @click="onNodeSubmit">
- </el-tab-pane>
- <!-- 扩展表单 -->
- <el-tab-pane label="扩展表单" name="2">
- <el-form :model="form" ref="extendFormRef" size="default" label-width="80px" class="pt15 pr15 pb15 pl15">
- v-for="(val, key) in node.from"
- :rules="[{ required: val.required, message: `${val.label}不能为空`, trigger: 'blur' }]"
- <el-select v-model="form[val.prop]" placeholder="请选择" v-if="val.type === 'select'" clearable :disabled="val.disabled">
- <el-checkbox-group v-model="form[val.prop]" v-if="val.type === 'checkbox'" :disabled="val.disabled">
- <el-checkbox label="美食推荐" name="type"></el-checkbox>
- <el-checkbox label="统计分析" name="type"></el-checkbox>
- </el-checkbox-group>
- <el-button class="mb15" @click="onExtendRefresh">
- <el-button type="primary" class="mb15" @click="onExtendSubmit" :loading="loading.extend">
- <!-- 图表可视化 -->
- <el-tab-pane label="图表可视化" name="3">
- <div style="height: 200px; width: 320px" ref="chartsMonitorRef"></div>
- </el-tabs>
-import { defineComponent, reactive, toRefs, ref, nextTick, getCurrentInstance } from 'vue';
-interface WorkflowDrawerNodeState {
- node: { [key: string]: any };
- nodeRules: any;
- tabsActive: string;
- loading: {
- extend: boolean;
- name: 'pagesWorkflowDrawerNode',
- const nodeFormRef = ref();
- const extendFormRef = ref();
- const chartsMonitorRef = ref();
- const state = reactive<WorkflowDrawerNodeState>({
- node: {},
- nodeRules: {
- id: [{ required: true, message: '请输入数据id', trigger: 'blur' }],
- nodeId: [{ required: true, message: '请输入节点id', trigger: 'blur' }],
- type: [{ required: true, message: '请输入类型', trigger: 'blur' }],
- left: [{ required: true, message: '请输入left坐标', trigger: 'blur' }],
- top: [{ required: true, message: '请输入top坐标', trigger: 'blur' }],
- icon: [{ required: true, message: '请输入icon图标', trigger: 'blur' }],
- name: [{ required: true, message: '请输入名称', trigger: 'blur' }],
- module: [],
- tabsActive: '1',
- extend: false,
- state.tabsActive = '1';
- state.node = data;
- // 节点编辑-重置
- const onNodeRefresh = () => {
- state.node.icon = '';
- state.node.name = '';
- // 节点编辑-保存
- const onNodeSubmit = () => {
- nodeFormRef.value.validate((valid: boolean) => {
- emit('submit', state.node);
- // 扩展表单-重置
- const onExtendRefresh = () => {
- extendFormRef.value.resetFields();
- // 扩展表单-保存
- const onExtendSubmit = () => {
- extendFormRef.value.validate((valid: boolean) => {
- state.loading.extend = true;
- state.loading.extend = false;
- ElMessage.success('保存成功');
- // 图表可视化-初始化
- const numsOne = [];
- const numsTwo = [];
- for (let i = 0; i < 7; i++) {
- numsOne.push(`${Math.floor(Math.random() * 52 + 10)}:${Math.floor(Math.random() * 52 + 1)}`);
- numsTwo.push(Math.floor(Math.random() * 52 + 1));
- right: 30,
- left: 50,
- data: numsOne,
- data: numsTwo,
- nodeFormRef,
- extendFormRef,
- chartsMonitorRef,
- onNodeRefresh,
- onExtendRefresh,
- onExtendSubmit,
-.workflow-drawer-node {
- ::v-deep {
- .el-tabs {
- box-shadow: unset;
- border: unset;
- .el-tabs__nav {
- .el-tabs__item {
- padding: unset;
- &:first-of-type.is-active {
- border-left-color: transparent;
- &:last-of-type.is-active {
- border-right-color: transparent;
- .el-tabs__content {
- padding: 0;
- height: calc(100vh - 90px);
- .el-tab-pane {
- <div class="workflow-tool-help">
- <el-dialog v-model="isShow" width="769px">
- <div v-drag="['.workflow-tool-help .el-dialog', '.workflow-tool-help .el-dialog__header']">使用帮助</div>
- <div>1、拖入:鼠标移入左侧导航中,鼠标形状改变时拖动到右侧网格状的视图中。</div>
- <div class="mt10">2、移动:鼠标移入到视图中的某个节点元素,鼠标形状改变时拖动改变位置。</div>
- <div class="mt10">3、连线:鼠标移入到视图中的某个节点元素的icon(图标),鼠标形状改变(变成"+"),按下鼠标左键进行拖线连接。</div>
- <div class="mt10">4、节点:鼠标移入到视图中的某个节点元素,点击鼠标右键可进行删除、编辑节点。</div>
- <div class="mt10 mb10">5、线条:鼠标移入到视图中的某个线条,线条颜色改变时,点击鼠标右键可进行删除、编辑线条。</div>
- name: 'pagesWorkflowToolHelp',
- // 打开弹窗
- const open = () => {
- // 关闭弹窗
@@ -1,76 +0,0 @@
- <div class="workflow-tool">
- <div class="pl15">{{ setToolTitle }}</div>
- <div class="workflow-tool-right">
- <div class="workflow-tool-icon" v-for="(v, k) in toolList" :key="k" :title="v.title" @click="onToolClick(v.fnName)">
-import { defineComponent, computed, reactive, toRefs } from 'vue';
- name: 'pagesWorkflowTool',
- toolList: [
- { icon: 'ele-Help', title: '帮助', fnName: 'help' },
- { icon: 'ele-Download', title: '下载', fnName: 'download' },
- { icon: 'ele-Check', title: '提交', fnName: 'submit' },
- { icon: 'ele-DocumentCopy', title: '复制', fnName: 'copy' },
- { icon: 'ele-Delete', title: '删除', fnName: 'del' },
- { icon: 'ele-FullScreen', title: '全屏', fnName: 'fullscreen' },
- // 设置 tool 标题
- const setToolTitle = computed(() => {
- let { globalTitle } = store.state.themeConfig.themeConfig;
- return `${globalTitle}工作流`;
- // 顶部工具栏
- const onToolClick = (fnName: string) => {
- emit('tool', fnName);
- setToolTitle,
- onToolClick,
-.workflow-tool {
- height: 35px;
- border-bottom: 1px solid var(--el-border-color-light, #ebeef5);
- .workflow-tool-right {
- justify-content: flex-end;
- &-icon {
- color: var(--next-bg-topBarColor);
- line-height: 35px;
- background: rgba(0, 0, 0, 0.04);
- animation: logoAnimation 0.3s ease-in-out;
@@ -1,684 +0,0 @@
- <div class="workflow-container">
- <div class="workflow-mask" v-if="isShow"></div>
- <div class="layout-view-bg-white flex" :style="{ height: `calc(100vh - ${setViewHeight}` }">
- <div class="workflow">
- <!-- 顶部工具栏 -->
- <Tool @tool="onToolClick" />
- <!-- 左侧导航区 -->
- <div class="workflow-content">
- <div id="workflow-left">
- :id="`left${key}`"
- v-for="(val, key) in leftNavList"
- :key="val.id"
- :style="{ height: val.isOpen ? 'auto' : '50px', overflow: 'hidden' }"
- class="workflow-left-id"
- <div class="workflow-left-title" @click="onTitleClick(val)">
- <span>{{ val.title }}</span>
- <SvgIcon :name="val.isOpen ? 'ele-ArrowDown' : 'ele-ArrowRight'" />
- <div class="workflow-left-item" v-for="(v, k) in val.children" :key="k" :data-name="v.name" :data-icon="v.icon" :data-id="v.id">
- <div class="workflow-left-item-icon">
- <SvgIcon :name="v.icon" class="workflow-icon-drag" />
- <div class="font10 pl5 name">{{ v.name }}</div>
- <!-- 右侧绘画区 -->
- <div id="workflow-right">
- v-for="(v, k) in jsplumbData.nodeList"
- :key="v.nodeId"
- :id="v.nodeId"
- :class="v.class"
- :style="{ left: v.left, top: v.top }"
- @click="onItemCloneClick(k)"
- @contextmenu.prevent="onContextmenu(v, k, $event)"
- <div class="workflow-right-box" :class="{ 'workflow-right-active': jsPlumbNodeIndex === k }">
- <!-- 节点右键菜单 -->
- <Contextmenu :dropdown="dropdownNode" ref="contextmenuNodeRef" @current="onCurrentNodeClick" />
- <!-- 线右键菜单 -->
- <Contextmenu :dropdown="dropdownLine" ref="contextmenuLineRef" @current="onCurrentLineClick" />
- <!-- 抽屉表单、线 -->
- <Drawer ref="drawerRef" @label="setLineLabel" @node="setNodeContent" />
- <!-- 顶部工具栏-帮助弹窗 -->
- <Help ref="helpRef" />
-import { defineComponent, toRefs, reactive, computed, onMounted, onUnmounted, nextTick, ref } from 'vue';
-import { ElMessage, ElMessageBox } from 'element-plus';
-import { jsPlumb } from 'jsplumb';
-import Sortable from 'sortablejs';
-import Tool from './component/tool/index.vue';
-import Help from './component/tool/help.vue';
-import Contextmenu from './component/contextmenu/index.vue';
-import Drawer from './component/drawer/index.vue';
-import { leftNavList } from './js/mock';
-import { jsplumbDefaults, jsplumbMakeSource, jsplumbMakeTarget, jsplumbConnect } from './js/config';
-interface NodeListState {
- id: string | number;
- nodeId: string | undefined;
- class: HTMLElement | string;
- left: number | string;
- top: number | string;
- icon: string;
- name: string;
-interface LineListState {
- sourceId: string;
- targetId: string;
-interface XyState {
- x: string | number;
- y: string | number;
-interface WorkflowState {
- leftNavList: any[];
- dropdownNode: XyState;
- dropdownLine: XyState;
- jsPlumb: any;
- jsPlumbNodeIndex: null | number;
- jsplumbDefaults: any;
- jsplumbMakeSource: any;
- jsplumbMakeTarget: any;
- jsplumbConnect: any;
- jsplumbData: {
- nodeList: Array<NodeListState>;
- lineList: Array<LineListState>;
- components: { Tool, Contextmenu, Drawer, Help },
- const contextmenuNodeRef = ref();
- const contextmenuLineRef = ref();
- const drawerRef = ref();
- const helpRef = ref();
- const state = reactive<WorkflowState>({
- leftNavList: [],
- dropdownNode: { x: '', y: '' },
- dropdownLine: { x: '', y: '' },
- jsPlumb: null,
- jsPlumbNodeIndex: null,
- jsplumbDefaults,
- jsplumbMakeSource,
- jsplumbMakeTarget,
- jsplumbConnect,
- nodeList: [],
- lineList: [],
- // 设置 view 的高度
- const setViewHeight = computed(() => {
- // 设置 宽度小于 768,不支持操
- const setClientWidth = () => {
- const clientWidth = document.body.clientWidth;
- clientWidth < 768 ? (state.isShow = true) : (state.isShow = false);
- // 左侧导航-数据初始化
- const initLeftNavList = () => {
- state.leftNavList = leftNavList;
- state.jsplumbData = {
- nodeList: [
- { nodeId: 'huej738hbji', left: '148px', top: '93px', class: 'workflow-right-clone', icon: 'iconfont icon-gongju', name: '引擎', id: '11' },
- nodeId: '52kcszzyxrd',
- left: '458px',
- top: '203px',
- class: 'workflow-right-clone',
- icon: 'iconfont icon-shouye_dongtaihui',
- name: '模版',
- id: '12',
- nodeId: 'nltskl6k4me',
- left: '164px',
- top: '350px',
- icon: 'iconfont icon-zhongduancanshuchaxun',
- name: '名称',
- id: '13',
- lineList: [
- { sourceId: 'huej738hbji', targetId: '52kcszzyxrd', label: '传送' },
- { sourceId: 'huej738hbji', targetId: 'nltskl6k4me', label: '' },
- // 左侧导航-初始化拖动
- const initSortable = () => {
- state.leftNavList.forEach((v, k) => {
- Sortable.create(document.getElementById(`left${k}`) as HTMLElement, {
- group: {
- name: 'vue-next-admin-1',
- pull: 'clone',
- put: false,
- animation: 0,
- sort: false,
- draggable: '.workflow-left-item',
- forceFallback: true,
- onEnd: function (evt: any) {
- const { name, icon, id } = evt.clone.dataset;
- const { layerX, layerY, clientX, clientY } = evt.originalEvent;
- const el = document.querySelector('#workflow-right') as HTMLElement;
- const { x, y, width, height } = el.getBoundingClientRect();
- if (clientX < x || clientX > width + x || clientY < y || y > y + height) {
- ElMessage.warning('请把节点拖入到画布中');
- // 节点id(唯一)
- const nodeId = Math.random().toString(36).substr(2, 12);
- // 处理节点数据
- const node = {
- nodeId,
- left: `${layerX - 40}px`,
- top: `${layerY - 15}px`,
- name,
- icon,
- id,
- // 右侧视图内容数组
- state.jsplumbData.nodeList.push(node);
- // 元素加载完毕时
- // 整个节点作为source或者target
- state.jsPlumb.makeSource(nodeId, state.jsplumbMakeSource);
- // // 整个节点作为source或者target
- state.jsPlumb.makeTarget(nodeId, state.jsplumbMakeTarget, jsplumbConnect);
- // 设置节点可以拖拽(此处为id值,非class)
- state.jsPlumb.draggable(nodeId, {
- containment: 'parent',
- stop: (el: any) => {
- state.jsplumbData.nodeList.forEach((v) => {
- if (v.nodeId === el.el.id) {
- // 节点x, y重新赋值,防止再次从左侧导航中拖拽节点时,x, y恢复默认
- v.left = `${el.pos[0]}px`;
- v.top = `${el.pos[1]}px`;
- // 初始化 jsPlumb
- const initJsPlumb = () => {
- (<any>jsPlumb).ready(() => {
- state.jsPlumb = (<any>jsPlumb).getInstance({
- detachable: false,
- Container: 'workflow-right',
- state.jsPlumb.fire('jsPlumbDemoLoaded', state.jsPlumb);
- // 导入默认配置
- state.jsPlumb.importDefaults(state.jsplumbDefaults);
- // 会使整个jsPlumb立即重绘。
- state.jsPlumb.setSuspendDrawing(false, true);
- // 初始化节点、线的链接
- initJsPlumbConnection();
- // 点击线弹出右键菜单
- state.jsPlumb.bind('contextmenu', (conn: any, originalEvent: MouseEvent) => {
- originalEvent.preventDefault();
- const { sourceId, targetId } = conn;
- const { clientX, clientY } = originalEvent;
- state.dropdownLine.x = clientX;
- state.dropdownLine.y = clientY;
- const v: any = state.jsplumbData.nodeList.find((v) => v.nodeId === targetId);
- const line: any = state.jsplumbData.lineList.find((v) => v.sourceId === sourceId && v.targetId === targetId);
- v.type = 'line';
- v.label = line.label;
- contextmenuLineRef.value.openContextmenu(v, conn);
- // 连线之前
- state.jsPlumb.bind('beforeDrop', (conn: any) => {
- const item = state.jsplumbData.lineList.find((v) => v.sourceId === sourceId && v.targetId === targetId);
- if (item) {
- ElMessage.warning('关系已存在,不可重复连接');
- return true;
- // 连线时
- state.jsPlumb.bind('connection', (conn: any) => {
- state.jsplumbData.lineList.push({
- sourceId,
- targetId,
- // 删除连线时回调函数
- state.jsPlumb.bind('connectionDetached', (conn: any) => {
- state.jsplumbData.lineList = state.jsplumbData.lineList.filter((line) => {
- if (line.sourceId == sourceId && line.targetId == targetId) {
- const initJsPlumbConnection = () => {
- // 节点
- state.jsPlumb.makeSource(v.nodeId, state.jsplumbMakeSource);
- state.jsPlumb.makeTarget(v.nodeId, state.jsplumbMakeTarget, jsplumbConnect);
- state.jsPlumb.draggable(v.nodeId, {
- // 线
- state.jsplumbData.lineList.forEach((v) => {
- state.jsPlumb.connect(
- source: v.sourceId,
- target: v.targetId,
- label: v.label,
- state.jsplumbConnect
- // 左侧导航-菜单标题点击
- const onTitleClick = (val: any) => {
- val.isOpen = !val.isOpen;
- // 右侧内容区-当前项点击
- const onItemCloneClick = (k: number) => {
- state.jsPlumbNodeIndex = k;
- // 右侧内容区-当前项右键菜单点击
- const onContextmenu = (v: any, k: number, e: MouseEvent) => {
- const { clientX, clientY } = e;
- state.dropdownNode.x = clientX;
- state.dropdownNode.y = clientY;
- v.type = 'node';
- v.label = '';
- let item: any = {};
- state.leftNavList.forEach((l) => {
- if (l.children) if (l.children.find((c: any) => c.id === v.id)) item = l.children.find((c: any) => c.id === v.id);
- v.from = item.form;
- contextmenuNodeRef.value.openContextmenu(v);
- // 右侧内容区-当前项右键菜单点击回调(节点)
- const onCurrentNodeClick = (item: any) => {
- const { contextMenuClickId, nodeId } = item;
- if (contextMenuClickId === 0) {
- const nodeIndex = state.jsplumbData.nodeList.findIndex((item) => item.nodeId === nodeId);
- state.jsplumbData.nodeList.splice(nodeIndex, 1);
- state.jsPlumb.removeAllEndpoints(nodeId);
- state.jsPlumbNodeIndex = null;
- } else if (contextMenuClickId === 1) {
- drawerRef.value.open(item);
- // 右侧内容区-当前项右键菜单点击回调(线)
- const onCurrentLineClick = (item: any, conn: any) => {
- const { contextMenuClickId } = item;
- const { endpoints } = conn;
- const intercourse: any = [];
- endpoints.forEach((v: any) => {
- intercourse.push({
- id: v.element.id,
- innerText: v.element.innerText,
- item.contact = `${intercourse[0].innerText}(${intercourse[0].id}) => ${intercourse[1].innerText}(${intercourse[1].id})`;
- if (contextMenuClickId === 0) state.jsPlumb.deleteConnection(conn);
- else if (contextMenuClickId === 1) drawerRef.value.open(item, conn);
- // 设置线的 label
- const setLineLabel = (obj: any) => {
- const { sourceId, targetId, label } = obj;
- const conn = state.jsPlumb.getConnections({
- source: sourceId,
- target: targetId,
- })[0];
- conn.setLabel(label);
- if (!label || label === '') {
- conn.addClass('workflow-right-empty-label');
- conn.removeClass('workflow-right-empty-label');
- conn.addClass('workflow-right-label');
- if (v.sourceId === sourceId && v.targetId === targetId) v.label = label;
- // 设置节点内容
- const setNodeContent = (obj: any) => {
- const { nodeId, name, icon } = obj;
- // 设置节点 name 与 icon
- if (v.nodeId === nodeId) {
- v.name = name;
- v.icon = icon;
- // 重绘
- // 顶部工具栏-当前项点击
- const onToolClick = (fnName: String) => {
- switch (fnName) {
- case 'help':
- onToolHelp();
- break;
- case 'download':
- onToolDownload();
- case 'submit':
- onToolSubmit();
- case 'copy':
- onToolCopy();
- case 'del':
- onToolDel();
- case 'fullscreen':
- onToolFullscreen();
- // 顶部工具栏-帮助
- const onToolHelp = () => {
- helpRef.value.open();
- // 顶部工具栏-下载
- const onToolDownload = () => {
- const { globalTitle } = store.state.themeConfig.themeConfig;
- const href = 'data:text/json;charset=utf-8,' + encodeURIComponent(JSON.stringify(state.jsplumbData, null, '\t'));
- const aLink = document.createElement('a');
- aLink.setAttribute('href', href);
- aLink.setAttribute('download', `${globalTitle}工作流.json`);
- aLink.click();
- aLink.remove();
- ElMessage.success('下载成功');
- // 顶部工具栏-提交
- const onToolSubmit = () => {
- // console.log(state.jsplumbData);
- ElMessage.success('数据提交成功');
- // 顶部工具栏-复制
- const onToolCopy = () => {
- copyText(JSON.stringify(state.jsplumbData));
- // 顶部工具栏-删除
- const onToolDel = () => {
- ElMessageBox.confirm('此操作将清空画布,是否继续?', '提示', {
- confirmButtonText: '清空',
- cancelButtonText: '取消',
- .then(() => {
- state.jsPlumb.removeAllEndpoints(v.nodeId);
- ElMessage.success('清空画布成功');
- .catch(() => {});
- // 顶部工具栏-全屏
- const onToolFullscreen = () => {
- store.dispatch('tagsViewRoutes/setCurrenFullscreen', true);
- onMounted(async () => {
- await initLeftNavList();
- initSortable();
- initJsPlumb();
- setClientWidth();
- window.addEventListener('resize', setClientWidth);
- window.removeEventListener('resize', setClientWidth);
- setViewHeight,
- setClientWidth,
- setLineLabel,
- setNodeContent,
- onTitleClick,
- onItemCloneClick,
- onContextmenu,
- onCurrentNodeClick,
- onCurrentLineClick,
- contextmenuNodeRef,
- contextmenuLineRef,
- drawerRef,
- helpRef,
-.workflow-container {
- .workflow {
- .workflow-content {
- height: calc(100% - 35px);
- #workflow-left {
- width: 220px;
- border-right: 1px solid var(--el-border-color-light, #ebeef5);
- ::v-deep(.el-collapse-item__content) {
- padding-bottom: 0;
- .workflow-left-title {
- height: 50px;
- border-top: 1px solid var(--el-border-color-light, #ebeef5);
- cursor: default;
- .workflow-left-item {
- width: calc(50% - 15px);
- cursor: move;
- margin: 0 0 10px 10px;
- .workflow-left-item-icon {
- padding: 5px 10px;
- border: 1px dashed transparent;
- background: var(--next-bg-color);
- i,
- .name {
- color: var(--el-text-color-secondary);
- border: 1px dashed var(--el-color-primary);
- background: var(--el-color-primary-light-9);
- border-radius: 5px;
- & .workflow-left-id:first-of-type {
- #workflow-right {
- background-image: linear-gradient(90deg, rgb(156 214 255 / 15%) 10%, rgba(0, 0, 0, 0) 10%),
- linear-gradient(rgb(156 214 255 / 15%) 10%, rgba(0, 0, 0, 0) 10%);
- background-size: 10px 10px;
- .workflow-right-clone {
- .workflow-right-box {
- min-width: 94.5px;
- border: 1px solid var(--el-border-color-light, #ebeef5);
- cursor: Crosshair;
- .workflow-right-active {
- ::v-deep(.jtk-overlay):not(.aLabel) {
- padding: 4px 10px;
- border: 1px solid var(--el-border-color-light, #ebeef5) !important;
- color: var(--el-text-color-secondary) !important;
- background: var(--el-color-white) !important;
- font-size: 10px;
- ::v-deep(.jtk-overlay.workflow-right-empty-label) {
- display: none;
- .workflow-mask {
- top: 0;
- bottom: 0;
- left: 0;
- content: '手机版不支持 jsPlumb 操作';
- background: rgba(255, 255, 255, 0.9);
@@ -1,99 +0,0 @@
-// jsplumb 默认配置
-export const jsplumbDefaults = {
- // 多个锚点 [源锚点,目标锚点]
- Anchors: [
- 'Top',
- 'TopCenter',
- 'TopRight',
- 'TopLeft',
- 'Right',
- 'RightMiddle',
- 'Bottom',
- 'BottomCenter',
- 'BottomRight',
- 'BottomLeft',
- 'Left',
- 'LeftMiddle',
- // 连线的容器id
- // 设置链接线的形状,如直线或者曲线之类的。anchor可以去设置锚点的位置。可选值"<Bezier|Flowchart|StateMachine|Straight>"
- Connector: ['Bezier', { curviness: 100 }],
- // 节点是否可以用鼠标拖动使其断开,默认为true。即用鼠标链接上的连线,也可以使用鼠标拖动让其断开。设置成false,可以让其拖动也不会自动断开
- ConnectionsDetachable: false,
- // 删除线的时候节点不删除
- DeleteEndpointsOnDetach: false,
- // 每当添加或以其他方式创建 Endpoint 并且 jsPlumb 尚未给出任何明确的 Endpoint 定义时将使用
- Endpoint: ['Blank', { Overlays: '' }],
- // 连接中源和目标端点的默认外观
- EndpointStyle: { fill: '#1879ffa1', outlineWidth: 1 },
- // jsPlumb 的内部日志记录是否打开
- LogEnabled: true,
- // 连接器的默认外观
- PaintStyle: {
- stroke: '#E0E3E7',
- strokeWidth: 1,
- outlineStroke: 'transparent',
- outlineWidth: 10,
- // 用于配置任何可拖动元素的默认选项jsPlumb.draggable
- DragOptions: { cursor: 'pointer', zIndex: 2000 },
- // 添加到连接器和端点的默认叠加层。已弃用:从 4.x 开始,将不支持此功能。并非所有叠加层都可以连接到连接器和端点。
- Overlays: [
- [
- 'Arrow',
- width: 10, // 箭头尾部的宽度
- length: 8, // 从箭头的尾部到头部的距离
- location: 1, // 位置,建议使用0~1之间
- direction: 1, // 方向,默认值为1(表示向前),可选-1(表示向后)
- foldback: 0.623, // 折回,也就是尾翼的角度,默认0.623,当为1时,为正三角
- 'Label',
- location: 0.5,
- cssClass: 'aLabel',
- // 默认渲染模式 svg、canvas
- RenderMode: 'svg',
- // 悬停状态下连接的默认外观
- HoverPaintStyle: { stroke: '#b0b2b5', strokeWidth: 1 },
- // 悬停状态下端点的默认外观
- EndpointHoverStyle: { fill: 'red' },
- // 端点和连接的默认范围。范围提供了对哪些端点可以连接到哪些其他端点的基本控制
- Scope: 'jsPlumb_DefaultScope',
-// 整个节点作为source或者target
-export const jsplumbMakeSource = {
- // 设置可以拖拽的类名,只要鼠标移动到该类名上的DOM,就可以拖拽连线
- filter: '.workflow-icon-drag',
- filterExclude: false,
- anchor: 'Continuous',
- // 是否允许自己连接自己
- allowLoopback: true,
- maxConnections: -1,
-export const jsplumbMakeTarget = {
- dropOptions: { hoverClass: 'ef-drop-hover' },
-// 连线参数
-export const jsplumbConnect = {
- isSource: true,
- isTarget: true,
- // 动态锚点、提供了4个方向 Continuous、AutoDefault
@@ -1,262 +0,0 @@
-// 左侧菜单导航数据
-export const leftNavList = [
- title: '工作流',
- icon: 'iconfont icon-shouye',
- isOpen: true,
- id: '1',
- name: '引擎',
- id: '11',
- form: [
- label: '客户姓名',
- placeholder: '请输入客户姓名',
- label: '性别',
- prop: 'sex',
- placeholder: '请选择性别',
- value: '0',
- label: '女',
- label: '男',
- label: '员工编号',
- prop: 'number',
- placeholder: '请输入员工编号',
- label: '办公电话',
- prop: 'mobile',
- placeholder: '请输入办公电话',
- label: '权限分配',
- prop: 'role',
- label: '编辑权限',
- label: '删除权限',
- type: 'checkbox',
- label: '模块选择',
- prop: 'module',
- placeholder: '请选择模块',
- label: '等级',
- prop: 'grade',
- placeholder: '请输入等级',
- label: '登记密码',
- prop: 'password',
- placeholder: '请输入登记密码',
- label: '数据表',
- prop: 'dataSheet',
- placeholder: '请输入数据表',
- label: '字段配置',
- prop: 'field',
- placeholder: '请输入字段配置',
- name: '版本',
- id: '14',
- label: '发布模板',
- prop: 'publish',
- placeholder: '请输入发布模板',
- icon: 'iconfont icon-bolangnengshiyanchang',
- name: '建模',
- id: '15',
- label: '内容模板',
- prop: 'content',
- placeholder: '请输入内容模板',
- icon: 'iconfont icon-xingqiu',
- name: '节点',
- id: '16',
- label: '活动名称6',
- prop: 'name16',
- title: '流程',
- id: '2',
- icon: 'iconfont icon-fuwenben',
- name: '实例',
- id: '21',
- label: '活动名称7',
- prop: 'name21',
- name: '轨迹',
- id: '22',
- label: '活动名称8',
- prop: 'name22',
- icon: 'iconfont icon-shangchuan',
- name: '数据',
- id: '23',
- label: '活动名称9',
- prop: 'name23',
- title: '任务',
- icon: 'iconfont icon-shuju',
- id: '3',
- icon: 'iconfont icon-icon-',
- name: '参与人',
- id: '31',
- label: '活动名称1',
- prop: 'name31',
- icon: 'iconfont icon-gerenzhongxin',
- name: '执行人',
- id: '32',
- label: '活动名称2',
- prop: 'name32',
- icon: 'iconfont icon-fangkuang',
- name: '工单',
- id: '33',
- label: '活动名称3',
- prop: 'name33',
@@ -1,47 +0,0 @@
- <div class="flex-margin color-primary">
- <div>paramsCommonDetails</div>
- <div class="mt10 mb10">路径:path: {{ params.path }}</div>
- <div>参数:query: {{ params.query }}</div>
-import { defineComponent, toRefs, reactive, computed, onMounted } from 'vue';
- params: {
- path: '',
- query: '',
- state.params = <any>route;
@@ -1,55 +0,0 @@
- <el-result icon="success" title="普通路由" subTitle="可 `开启 TagsView 共用` 进行单标签测试">
- <template #extra>
- <el-input v-model="value" placeholder="请输入路由参数id值" clearable></el-input>
- <el-button type="primary" size="default" class="mt15" @click="onGoDetailsClick">
- <SvgIcon name="iconfont icon-putong" />
- 普通路由传参
- </el-result>
-import { toRefs, reactive, computed, defineComponent } from 'vue';
- value: '',
- // 跳转到详情
- const onGoDetailsClick = () => {
- path: '/params/common/details',
- query: { id: state.value, name: 'vue-next-admin' },
- state.value = '';
- onGoDetailsClick,
- <div>paramsDynamicDetails</div>
- <div>参数:params: {{ params.params }}</div>
- params: '',
- <el-result icon="warning" title="动态路由" subTitle="可 `开启 TagsView 共用` 进行单标签测试">
- <SvgIcon name="iconfont icon-dongtai" />
- 动态路由传参
-import { defineComponent, toRefs, reactive, computed } from 'vue';
- // name 值为路由中的 name
- t: 'vue-next-admin',
- id: state.value,
@@ -1,498 +0,0 @@
- <el-card shadow="hover" header="正则验证(一些项目中常用的正则)">
- <el-form :model="ruleForm" :rules="rules" class="tools-warp-form" size="default" label-position="top">
- <el-form-item label="验证百分比(不可以小数):" prop="a22">
- <div class="tools-warp-form-msg">验证可以输入大于0小于100的数字</div>
- <el-input v-model="ruleForm.a22" @input="onVerifyNumberPercentage($event)" placeholder="请输入数字进行测试">
- <template #append> % </template>
- <el-form-item label="验证百分比(可以小数):" prop="a23" class="mt20">
- <el-input v-model="ruleForm.a23" @input="onVerifyNumberPercentageFloat($event)" placeholder="请输入数字进行测试">
- <el-form-item label="小数或整数:" prop="a1" class="mt20">
- <div class="tools-warp-form-msg">
- 验证可以输入小数或整数,0 开始, . 只能出现一次,保留小数点后保留2位小数。(负数时,模拟拼接负号给后台)。
- <el-input v-model="ruleForm.a1" @input="onVerifyNumberIntegerAndFloat($event)" placeholder="请输入小数或整数进行测试"> </el-input>
- <el-form-item label="正整数:" prop="a2" class="mt20">
- <div class="tools-warp-form-msg">验证只可以输入正整数,0 开始后面将不可以输入。</div>
- <el-input v-model="ruleForm.a2" @input="onVerifiyNumberInteger($event)" placeholder="请输入整数进行测试"> </el-input>
- <el-form-item label="去掉中文及空格:" prop="a3" class="mt20">
- <div class="tools-warp-form-msg">验证不可以输入空格与中文。</div>
- <el-input v-model="ruleForm.a3" @input="onVerifyCnAndSpace($event)" placeholder="请输入内容进行测试"> </el-input>
- <el-form-item label="去掉英文及空格:" prop="a4" class="mt20">
- <div class="tools-warp-form-msg">验证不可以输入空格与英文。</div>
- <el-input v-model="ruleForm.a4" @input="onVerifyEnAndSpace($event)" placeholder="请输入内容进行测试"> </el-input>
- <el-form-item label="禁止输入空格:" prop="a5" class="mt20">
- <div class="tools-warp-form-msg">验证不可以输入空格。</div>
- <el-input v-model="ruleForm.a5" @input="onVerifyAndSpace($event)" placeholder="请输入内容进行测试"> </el-input>
- <el-form-item label="金额用 `,` 区分开:" prop="a6" class="mt20">
- <div class="tools-warp-form-msg">金额添加 `,` 进行区分,便于阅读。{{ ruleForm.a6 }}</div>
- <el-input v-model="ruleForm.a6" @input="onVerifyNumberComma($event)" placeholder="请输入金额进行测试"> </el-input>
- <el-form-item label="匹配文字变色(搜索时):" prop="a7" class="mt20">
- <div class="tools-warp-form-msg">示例:<span v-html="text"></span></div>
- <el-input v-model="ruleForm.a7" @input="onVerifyTextColor($event)" placeholder="请输入示例中的部分文字"> </el-input>
- <el-form-item label="数字转中文大写:" prop="a8" class="mt20">
- 验证数字转成中文的大写。<span class="tools-warp-form-msg-red">{{ cnText }}</span>
- <el-input v-model="ruleForm.a8" @input="onVerifyNumberCnUppercase($event)" placeholder="请输入金额进行测试"> </el-input>
- <el-form-item label="手机号码:" prop="a9" class="mt20">
- 验证手机号码 (true: 正确,false: 不正确)。<span class="tools-warp-form-msg-red">{{ phone }}</span>
- <el-input v-model="ruleForm.a9" @input="onVerifyPhone($event)" placeholder="请输入手机号进行测试" maxlength="11"> </el-input>
- <el-form-item label="国内电话号码:" prop="a10" class="mt20">
- 验证国内电话号码 (true: 正确,false: 不正确)。<span class="tools-warp-form-msg-red">{{ telePhone }}</span>
- <el-input v-model="ruleForm.a10" @input="onVerifyTelPhone($event)" placeholder="请输入国内电话号码进行测试" maxlength="12"> </el-input>
- <el-form-item label="登录账号:" prop="a11" class="mt20">
- 验证登录账号是否正确。字母开头,允许5-16字节,允许字母数字下划线 (true: 正确,false: 不正确)。<span class="tools-warp-form-msg-red">{{
- account
- <el-input v-model="ruleForm.a11" @input="onVerifyAccount($event)" placeholder="请输入账号进行测试" maxlength="16"> </el-input>
- <el-form-item label="密码:" prop="a12" class="mt20">
- 验证密码是否正确。以字母开头,长度在6~16之间,只能包含字母、数字和下划线 (true: 正确,false: 不正确)。<span
- class="tools-warp-form-msg-red"
- >{{ password }}</span
- <el-input v-model="ruleForm.a12" @input="onVerifyPassword($event)" placeholder="请输入密码进行测试" maxlength="16"> </el-input>
- <el-form-item label="强密码:" prop="a13" class="mt20">
- 验证强密码是否正确。字母+数字+特殊字符,长度在6-16之间 (true: 正确,false: 不正确)。<span class="tools-warp-form-msg-red">{{
- passwordPowerful
- <el-input v-model="ruleForm.a13" @input="onVerifyPasswordPowerful($event)" placeholder="请输入密码进行测试" maxlength="16"> </el-input>
- <el-form-item label="密码强度:" prop="a14" class="mt20">
- 验证密码强度。返回 强、中、弱。(弱:纯数字,纯字母,纯特殊字符,中:字母+数字,字母+特殊字符,数字+特殊字符,强:字母+数字+特殊字符)<span
- >{{ passwordStrength }}</span
- <el-input v-model="ruleForm.a14" @input="onVerifyPasswordStrength($event)" placeholder="请输入密码进行测试" maxlength="16"> </el-input>
- <el-form-item label="IP地址:" prop="a15" class="mt20">
- 验证IP地址是否正确。(true: 正确,false: 不正确)。<span class="tools-warp-form-msg-red">{{ iPAddress }}</span>
- <el-input v-model="ruleForm.a15" @input="onVerifyIPAddress($event)" placeholder="请输入IP地址进行测试"> </el-input>
- <el-form-item label="邮箱:" prop="a16" class="mt20">
- 验证邮箱是否正确。(true: 正确,false:不正确)。<span class="tools-warp-form-msg-red">{{ email }}</span>
- <el-input v-model="ruleForm.a16" @input="onVerifyEmail($event)" placeholder="请输入邮箱进行测试"> </el-input>
- <el-form-item label="身份证:" prop="a17" class="mt20">
- 验证身份证是否正确。(true: 正确,false:不正确)。<span class="tools-warp-form-msg-red">{{ idCard }}</span>
- <el-input v-model="ruleForm.a17" @input="onVerifyIDCard($event)" placeholder="请输入身份证进行测试" maxlength="18"> </el-input>
- <el-form-item label="姓名:" prop="a18" class="mt20">
- 验证姓名是否正确,包括少数民族名字。(true: 正确,false:不正确)。<span class="tools-warp-form-msg-red">{{ fullName }}</span>
- <el-input v-model="ruleForm.a18" @input="onVerifyFullName($event)" placeholder="请输入姓名进行测试"> </el-input>
- <el-form-item label="邮政编码:" prop="a19" class="mt20">
- 验证邮政编码是否正确,不能以 0 开始。(true: 正确,false:不正确)。<span class="tools-warp-form-msg-red">{{ postalCode }}</span>
- <el-input v-model="ruleForm.a19" @input="onVerifyPostalCode($event)" placeholder="请输入邮政编码进行测试" maxlength="6"> </el-input>
- <el-form-item label="url:" prop="a20" class="mt20">
- 验证url是否正确。(true: 正确,false:不正确)。<span class="tools-warp-form-msg-red">{{ url }}</span>
- <el-input v-model="ruleForm.a20" @input="onVerifyUrl($event)" placeholder="请输入内容进行测试"> </el-input>
- <el-form-item label="车牌号:" prop="a21" class="mt20">
- 验证车牌号是否正确。(true: 正确,false:不正确)。<span class="tools-warp-form-msg-red">{{ carNum }}</span>
- <el-input v-model="ruleForm.a21" @input="onVerifyCarNum($event)" placeholder="请输入车牌号进行测试"> </el-input>
-import {
- verifyNumberPercentage,
- verifyNumberPercentageFloat,
- verifyNumberIntegerAndFloat,
- verifiyNumberInteger,
- verifyCnAndSpace,
- verifyEnAndSpace,
- verifyAndSpace,
- verifyNumberComma,
- verifyTextColor,
- verifyNumberCnUppercase,
- verifyPhone,
- verifyTelPhone,
- verifyAccount,
- verifyPassword,
- verifyPasswordPowerful,
- verifyPasswordStrength,
- verifyIPAddress,
- verifyEmail,
- verifyIdCard,
- verifyFullName,
- verifyPostalCode,
- verifyUrl,
- verifyCarNum,
-} from '/@/utils/toolsValidate';
- text: '世间美好,与你环环相扣,祝你开心每一天!',
- phone: false,
- cnText: '',
- telePhone: false,
- account: false,
- password: false,
- passwordPowerful: false,
- passwordStrength: '',
- iPAddress: false,
- email: false,
- idCard: false,
- fullName: false,
- postalCode: false,
- url: false,
- carNum: false,
- * 变量名为了方便,随便取了,
- * 实际中,按正常程序进行命名
- ruleForm: {
- a9: '',
- a10: '',
- a11: '',
- a12: '',
- a13: '',
- a14: '',
- a15: '',
- a16: '',
- a17: '',
- a18: '',
- a19: '',
- a20: '',
- a21: '',
- a22: '',
- a23: '',
- a1: [
- message: '请输入小数或整数进行测试',
- trigger: 'change',
- a2: [
- message: '请输入正整数进行测试',
- a3: [{ required: true, message: '请输入内容进行测试', trigger: 'change' }],
- a4: [{ required: true, message: '请输入内容进行测试', trigger: 'change' }],
- a5: [{ required: true, message: '请输入内容进行测试', trigger: 'change' }],
- a6: [
- a7: [{ required: true, message: '请输入内容进行测试', trigger: 'change' }],
- a8: [{ required: true, message: '请输入金额进行测试', trigger: 'change' }],
- a9: [
- message: '请输入手机号进行测试',
- a10: [
- message: '请输入国内电话号码进行测试',
- a11: [{ required: true, message: '请输入账号进行测试', trigger: 'change' }],
- a12: [{ required: true, message: '请输入密码进行测试', trigger: 'change' }],
- a13: [{ required: true, message: '请输入密码进行测试', trigger: 'change' }],
- a14: [{ required: true, message: '请输入密码进行测试', trigger: 'change' }],
- a15: [
- message: '请输入IP地址进行测试',
- a16: [{ required: true, message: '请输入邮箱进行测试', trigger: 'change' }],
- a17: [
- message: '请输入身份证进行测试',
- a18: [{ required: true, message: '请输入姓名进行测试', trigger: 'change' }],
- a19: [
- message: '请输入邮政编码进行测试',
- a20: [{ required: true, message: '请输入内容进行测试', trigger: 'change' }],
- a21: [
- message: '请输入车牌号进行测试',
- a22: [{ required: true, message: '请输入数字进行测试', trigger: 'change' }],
- a23: [{ required: true, message: '请输入数字进行测试', trigger: 'change' }],
- // 验证百分比(不可以小数)
- const onVerifyNumberPercentage = (val: string) => {
- state.ruleForm.a22 = verifyNumberPercentage(val);
- // 验证百分比(可以小数)
- const onVerifyNumberPercentageFloat = (val: string) => {
- state.ruleForm.a23 = verifyNumberPercentageFloat(val);
- // 小数或整数
- const onVerifyNumberIntegerAndFloat = (val: string) => {
- state.ruleForm.a1 = verifyNumberIntegerAndFloat(val);
- // 正整数
- const onVerifiyNumberInteger = (val: string) => {
- state.ruleForm.a2 = verifiyNumberInteger(val);
- // 去掉中文及空格
- const onVerifyCnAndSpace = (val: string) => {
- state.ruleForm.a3 = verifyCnAndSpace(val);
- // 去掉英文及空格
- const onVerifyEnAndSpace = (val: string) => {
- state.ruleForm.a4 = verifyEnAndSpace(val);
- // 禁止输入空格
- const onVerifyAndSpace = (val: string) => {
- state.ruleForm.a5 = verifyAndSpace(val);
- // 金额用 `,` 区分开
- const onVerifyNumberComma = (val: string) => {
- state.ruleForm.a6 = verifyNumberComma(val);
- // 匹配文字变色(搜索时)
- const onVerifyTextColor = (val: string) => {
- state.ruleForm.a7 = verifyAndSpace(val);
- if (state.ruleForm.a7 === '') state.text = `世间美好,与你环环相扣,祝你开心每一天!`;
- else state.text = verifyTextColor(state.ruleForm.a7, state.text);
- // 数字转中文大写
- const onVerifyNumberCnUppercase = (val: string) => {
- state.ruleForm.a8 = verifyNumberIntegerAndFloat(val);
- if (state.ruleForm.a8 === '') state.cnText = '';
- else state.cnText = verifyNumberCnUppercase(state.ruleForm.a8);
- // 手机号码
- const onVerifyPhone = (val: string) => {
- state.phone = verifyPhone(val);
- // 国内电话号码
- const onVerifyTelPhone = (val: string) => {
- state.telePhone = verifyTelPhone(val);
- // 登录账号
- const onVerifyAccount = (val: string) => {
- state.ruleForm.a11 = verifyCnAndSpace(val);
- state.account = verifyAccount(state.ruleForm.a11);
- // 密码
- const onVerifyPassword = (val: string) => {
- state.ruleForm.a12 = verifyCnAndSpace(val);
- state.password = verifyPassword(state.ruleForm.a12);
- // 强密码
- const onVerifyPasswordPowerful = (val: string) => {
- state.ruleForm.a13 = verifyCnAndSpace(val);
- state.passwordPowerful = verifyPasswordPowerful(state.ruleForm.a13);
- // 密码强度
- const onVerifyPasswordStrength = (val: string) => {
- state.ruleForm.a14 = verifyCnAndSpace(val);
- state.passwordStrength = verifyPasswordStrength(state.ruleForm.a14);
- // IP地址
- const onVerifyIPAddress = (val: string) => {
- state.iPAddress = verifyIPAddress(val);
- // 邮箱
- const onVerifyEmail = (val: string) => {
- state.ruleForm.a16 = verifyCnAndSpace(val);
- state.email = verifyEmail(state.ruleForm.a16);
- // 身份证
- const onVerifyIDCard = (val: string) => {
- state.ruleForm.a17 = verifyCnAndSpace(val);
- state.idCard = verifyIdCard(state.ruleForm.a17);
- // 姓名
- const onVerifyFullName = (val: string) => {
- state.ruleForm.a18 = verifyAndSpace(val);
- state.fullName = verifyFullName(state.ruleForm.a18);
- // 邮政编码
- const onVerifyPostalCode = (val: string) => {
- state.ruleForm.a19 = verifiyNumberInteger(val);
- state.postalCode = verifyPostalCode(state.ruleForm.a19);
- // url
- const onVerifyUrl = (val: string) => {
- state.ruleForm.a20 = verifyAndSpace(val);
- state.url = verifyUrl(state.ruleForm.a20);
- // 车牌号
- const onVerifyCarNum = (val: string) => {
- state.ruleForm.a21 = verifyAndSpace(val);
- state.carNum = verifyCarNum(state.ruleForm.a21);
- onVerifyNumberPercentage,
- onVerifyNumberPercentageFloat,
- onVerifyNumberIntegerAndFloat,
- onVerifiyNumberInteger,
- onVerifyCnAndSpace,
- onVerifyEnAndSpace,
- onVerifyAndSpace,
- onVerifyNumberComma,
- onVerifyTextColor,
- onVerifyNumberCnUppercase,
- onVerifyPhone,
- onVerifyTelPhone,
- onVerifyAccount,
- onVerifyPassword,
- onVerifyPasswordPowerful,
- onVerifyPasswordStrength,
- onVerifyIPAddress,
- onVerifyEmail,
- onVerifyIDCard,
- onVerifyFullName,
- onVerifyPostalCode,
- onVerifyUrl,
- onVerifyCarNum,
-.tools-warp-form {
- ::v-deep(.el-form-item--small.el-form-item) {
- margin-bottom: 0 !important;
- .tools-warp-form-msg {
- .tools-warp-form-msg-red {
- color: red;
- .tools-warp-form-msg + div {
@@ -1,1274 +0,0 @@
- <div class="visualizing-demo1">
- <!-- 地图 -->
- <div id="visualizingDemo1" style="height: 100%"></div>
- <div class="visualizing-container">
- <!-- 头部 -->
- <div class="visualizing-container-head">
- <div class="visualizing-container-head-left">
- <div class="visualizing-container-head-left-text">
- <div class="visualizing-container-head-left-text-box">{{ time.txt }}</div>
- <div class="visualizing-container-head-center">
- <div class="visualizing-container-head-center-box">
- <div class="visualizing-container-head-center-maintitle">深圳市xxx软件科技有限公司</div>
- <div class="visualizing-container-head-center-subtitle">旅游经济</div>
- <div class="visualizing-container-head-right">
- <div class="visualizing-container-head-right-text">
- <div class="visualizing-container-head-right-text-box">🌤 多云转晴东南风 26~30℃</div>
- <!-- 图表左侧 -->
- <div class="visualizing-container-content-left">
- <div class="visualizing-container-content-left-flex">
- <div class="visualizing-container-title">
- <i class="el-icon-s-shop"></i>
- <span>产业概况</span>
- <hr class="visualizing-container-title-colorful" />
- <div ref="visualizingContentLeftTop" style="height: 100%"></div>
- <i class="el-icon-s-promotion"></i>
- <span>A级风景区对比</span>
- <div ref="visualizingContentLeftBottom" style="height: 100%"></div>
- <!-- 图表中间 -->
- <div class="visualizing-container-content-center">
- <div class="visualizing-container-content-center-bottom">
- <div class="visualizing-container-content-center-bottom-flex">
- <i class="el-icon-s-custom"></i>
- <span>游客过夜情况</span>
- <div ref="visualizingContentCenterTop" style="height: 100%"></div>
- <i class="el-icon-s-flag"></i>
- <span>游客驻留时长</span>
- <div ref="visualizingContentCenterBottom" style="height: 100%"></div>
- <!-- 图表右侧 -->
- <div class="visualizing-container-content-right">
- <div class="visualizing-container-content-right-flex">
- <i class="el-icon-s-marketing"></i>
- <span>当日游客趋势分析</span>
- <div ref="visualizingContentRightTop" style="height: 100%"></div>
- <i class="el-icon-s-data"></i>
- <span>当月游客趋势分析</span>
- <div ref="visualizingContentRightBottom" style="height: 100%"></div>
-import { toRefs, reactive, onMounted, onUnmounted, getCurrentInstance, defineComponent } from 'vue';
-import { echartsMapList, echartsMapData, echartsMapImgs } from './mock/demo1';
-interface Demo1State {
- echartsMapList: any;
- echartsMapData: any;
- echartsMapImgs: any;
- time: any;
- myCharts: any[];
- const state = reactive<Demo1State>({
- echartsMapImgs,
- state.time.txt = formatDate(new Date(), 'YYYY-mm-dd HH:MM:SS WWW QQQQ ZZZ');
- // 初始化 echartsMap(地图上的点)
- const myChart = echarts.init(<HTMLElement>document.getElementById('visualizingDemo1'));
- formatter(params: any) {
- // 自定义鼠标放入样式
- let item = state.echartsMapImgs.find((v: any) => v.name === params.name);
- let html = `<div style="width: 240px">
- <div style="display: flex; align-items: center">
- <img src="${item?.url}" style="width: 50px; height: 50px; border-radius: 100%; position: relative; border: 4px solid #ffffff; margin-left: -4px" />
- style="
- background: #51c1ff;
- height: 32px;
- margin-left: -14px;
- padding-left: 20px;
- color: #fff;
- ${item?.name}
- <div style="margin-top: 10px; font-size: 12px">
- <div style="width: 61px"><i class="el-icon-location-information" style="margin-right: 5px"></i>地址:</div>
- <div style="flex: 1; white-space: pre-wrap; word-break: break-all; margin-top: 5px; color: #333">${item?.add}</div>
- <div style="width: 61px"><i class="el-icon-chat-dot-round" style="margin-right: 5px"></i>概括:</div>
- <div style="flex: 1; white-space: pre-wrap; word-break: break-all; margin-top: 5px; color: #333">${item?.dec}</div>
- </div>`;
- return html;
- color: ['#ea7ccc'],
- center: [114.064524, 22.549225],
- zoom: 11,
- name: '门票收入',
- shadowBlur: 100,
- state.myCharts.push(myChart);
- // 地图
- const map = (<any>myChart).getModel().getComponent('bmap').getBMap();
- // BMAP_NORMAL_MAP :此地图类型展示普通街道视图
- // BMAP_PERSPECTIVE_MAP :此地图类型展示透视图像视图。(这个还不会用)
- // BMAP_SATELLITE_MAP:卫星地图 (没有坐标, 绿绿的一片的卫星地图)
- // BMAP_HYBRID_MAP:混合地图 (既有坐标,也是绿绿的一片的卫星地图)
- // eslint-disable-next-line no-undef
- map.setMapType(BMAP_SATELLITE_MAP);
- let bdary = new BMap.Boundary();
- // 获取行政区域
- bdary.get('深圳', function (rs: any) {
- // 行政区域的点有多少个
- let count = rs.boundaries.length;
- for (let i = 0; i < count; i++) {
- let ply = new BMap.Polygon(rs.boundaries[i], {
- // 设置多边形边线线粗
- strokeWeight: 4,
- // 设置多边形边线透明度0-1
- strokeOpacity: 1,
- // 设置多边形边线样式为实线或虚线,取值 solid 或 dashed
- StrokeStyle: 'dashed',
- // 设置多边形边线颜色
- strokeColor: '#febb50',
- // 设置多边形填充颜色
- fillColor: '',
- // 建立多边形覆盖物
- // 添加覆盖物
- map.addOverlay(ply);
- // 调整视野
- map.setViewport(ply.getPath());
- // 初始化地图,设置中心点坐标和地图级别
- // new BMap.Point('深圳市', 11)
- map.centerAndZoom(new BMap.Point(114.064524, 22.549225), 11);
- // 产业概况
- const initVisualizingContentLeftTop = () => {
- const myChart = echarts.init(proxy.$refs.visualizingContentLeftTop);
- right: 0,
- bottom: 50,
- data: ['1月', '2月', '3月', '4月', '5月', '6月'],
- axisLine: {
- lineStyle: {
- color: 'rgba(22, 207, 208, 0.1)',
- width: 1,
- axisTick: {
- axisLabel: {
- color: '#16cfd0',
- yAxis: [
- name: '价格',
- splitLine: {
- color: 'rgba(22, 207, 208, 0.3)',
- splitArea: {
- color: 'rgba(22, 207, 208, 0.02)',
- nameTextStyle: {
- name: '预购队列',
- data: [200, 85, 112, 275, 305, 415],
- name: '最新成交价',
- data: [50, 85, 22, 155, 170, 25],
- color: '#febb50',
- // A级风景区对比
- const initVisualizingContentLeftBottom = () => {
- const myChart = echarts.init(proxy.$refs.visualizingContentLeftBottom);
- right: 10,
- bottom: 40,
- data: ['1月', '2月', '3月', '4月', '5月'],
- interval: 0,
- fontSize: 10,
- name: '销量',
- name: '客流',
- smooth: true,
- width: 0,
- opacity: 0.8,
- color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
- offset: 0,
- color: 'rgba(128, 255, 165)',
- offset: 1,
- color: 'rgba(1, 191, 236)',
- ]),
- focus: 'series',
- data: [140, 232, 101, 264, 90],
- name: '天数',
- color: 'rgba(0, 221, 255)',
- color: 'rgba(77, 119, 255)',
- data: [120, 282, 111, 234, 220],
- // 游客过夜情况
- const initVisualizingContentCenterTop = () => {
- const myChart = echarts.init(proxy.$refs.visualizingContentCenterTop);
- const min = 100;
- const max = 1000;
- bottom: 66,
- left: 38,
- axisPointer: {
- type: 'shadow',
- xAxis: [
- data: ['地区', '地区', '地区', '地区', '地区', '地区', '地区', '地区', '地区', '地区'],
- fontSize: 9,
- rotate: -45,
- nameGap: 25,
- barWidth: 15,
- normal: {
- color: '#de682e',
- color: '#ecc232',
- position: 'top',
- formatter: function (param: any) {
- if (param.value == max || param.value == min) {
- return '';
- return param.value;
- color: 'rgba(22, 207, 208, 0.8)',
- markPoint: {
- symbolSize: 30,
- color: '#ffffff',
- { name: '年最低', value: min, xAxis: 0, yAxis: 100 },
- { name: '年最高', value: max, xAxis: 9, yAxis: 1000 },
- data: [100, 200, 300, 400, 500, 600, 700, 800, 900, 1000],
- // 游客驻留时长
- const initVisualizingContentCenterBottom = () => {
- const myChart = echarts.init(proxy.$refs.visualizingContentCenterBottom);
- top: 26,
- left: 45,
- name: '已完成',
- stack: 'total',
- barWidth: 12,
- labelLine: {
- color: new echarts.graphic.LinearGradient(0, 0, 1, 0, [
- { offset: 0, color: 'rgba(7,165,255,0.2)' },
- { offset: 1, color: 'rgba(7,165,255,1)' },
- name: '进行中',
- { offset: 0, color: 'rgba(41,244,236,0)' },
- { offset: 1, color: 'rgba(41,244,236,1)' },
- dataset: {
- source: [
- { status: '已签收', value1: 33, value2: 93 },
- { status: '配送中', value1: 53, value2: 32 },
- { status: '已出库', value1: 78, value2: 65 },
- { status: '采购中', value1: 12, value2: 35 },
- { status: '接单中', value1: 90, value2: 52 },
- // 当日游客趋势分析
- const initVisualizingContentRightTop = () => {
- const myChart = echarts.init(proxy.$refs.visualizingContentRightTop);
- left: 20,
- color: 'rgba(22, 207, 208, 0.5)',
- name: '亿元',
- name: '同比',
- formatter: '{value}%',
- name: '销售水量',
- yAxisIndex: 1,
- showAllSymbol: true,
- symbol: 'circle',
- color: '#058cff',
- color: 'rgba(5,140,255, 0.2)',
- data: [4.2, 3.8, 4.8, 3.5, 2.9, 2.8],
- name: '主营业务',
- color: '#00FFE3',
- color: '#4693EC',
- // 当月游客趋势分析
- const initVisualizingContentRightBottom = () => {
- const myChart = echarts.init(proxy.$refs.visualizingContentRightBottom);
- name: '人数(万)',
- data: [20, 15, 40, 55, 65, 85],
- color: '#EA7CCC',
- data: [30, 45, 65, 85, 60, 105],
- color: '#FAC958',
- state.myCharts[i].resize();
- await initEchartsMap();
- await initVisualizingContentLeftTop();
- await initVisualizingContentLeftBottom();
- await initVisualizingContentCenterTop();
- await initVisualizingContentCenterBottom();
- await initVisualizingContentRightTop();
- await initVisualizingContentRightBottom();
- await initEchartsResize();
-// 左右两侧图表宽度
-$lrWidth: 288px;
-// 中部图表高度
-$cheight: 240px;
-// 标题宽度
-$titleWidth: 240px;
-.visualizing-demo1 {
- ::v-deep(.BMap_cpyCtrl) {
- ::v-deep(.anchorBL) {
- .visualizing-container {
- .visualizing-container-head {
- height: 60px;
- color: #ffffff;
- background: linear-gradient(to bottom, rgba(0, 0, 0, 0.9), rgba(0, 0, 0, 0.3), rgba(0, 0, 0, 0.02));
- z-index: 3;
- .visualizing-container-head-left {
- .visualizing-container-head-left-text {
- opacity: 0.6;
- width: 93%;
- padding-left: 15px;
- transform: translateY(-100%);
- .visualizing-container-head-left-text-box {
- height: 20px;
- background: linear-gradient(to right, rgba(22, 207, 208, 0.1), rgba(22, 207, 208, 0.3));
- top: -1px;
- transform: skew(30deg);
- height: 3px;
- background: linear-gradient(-90deg, #16cfd0, transparent);
- right: 33px;
- opacity: 0.8;
- top: 41px;
- right: -2px;
- .visualizing-container-head-center {
- padding: 0 60px;
- background: radial-gradient(rgba(0, 0, 0, 0.7) 5%, rgba(0, 0, 0, 0.5) 15%, rgba(0, 0, 0, 0.02) 70%);
- .visualizing-container-head-center-box {
- .visualizing-container-head-center-maintitle {
- min-width: 21%;
- height: 29px;
- top: 2px;
- opacity: 0.3;
- border-top: 1px solid #16cfd0;
- left: -70px;
- border-right: 2px solid #16cfd0;
- transform: skew(50deg);
- right: -70px;
- border-left: 2px solid #16cfd0;
- transform: skew(-50deg);
- .visualizing-container-head-center-subtitle {
- min-width: 35%;
- height: 23px;
- border-bottom: 2px solid #16cfd0;
- left: -33px;
- border-left: 3px solid #16cfd0;
- border-image: linear-gradient(to right, #16cfd0, rgba(22, 207, 208, 0.02)) 1 10;
- right: -33px;
- border-right: 3px solid #16cfd0;
- border-image: linear-gradient(to left, rgba(22, 207, 208, 1), rgba(22, 207, 208, 0.02)) 1 10;
- width: 36px;
- height: 17px;
- top: -8px;
- border-bottom: 2px solid rgba(22, 207, 208, 0.7);
- transform: skew(32deg);
- left: -89px;
- border-left: 3px solid rgba(22, 207, 208, 0.7);
- transform: skew(-32deg);
- right: -89px;
- border-right: 3px solid rgba(22, 207, 208, 0.7);
- min-width: 50%;
- height: 15px;
- bottom: 0px;
- border-bottom: 4px solid #16cfd0;
- transform: skew(60deg);
- left: 13px;
- border-left: 5px solid #16cfd0;
- transform: skew(-60deg);
- right: 13px;
- border-right: 5px solid #16cfd0;
- .visualizing-container-head-right {
- .visualizing-container-head-right-text {
- float: right;
- padding-right: 15px;
- .visualizing-container-head-right-text-box {
- background: linear-gradient(to left, rgba(22, 207, 208, 0.1), rgba(22, 207, 208, 0.3));
- transform: skew(-30deg);
- background: linear-gradient(90deg, #16cfd0, transparent);
- left: 33px;
- left: -2px;
- .visualizing-container-title {
- max-width: $titleWidth;
- border-bottom: 1px solid #ffffff;
- border-image: linear-gradient(to right, #ffffff, rgba(22, 207, 208, 0.02)) 1 10;
- padding-right: 5px;
- color: orange;
- width: 1px;
- height: 10px;
- background: linear-gradient(to top, #ffffff, rgba(255, 255, 255, 0.5));
- .visualizing-container-title-colorful {
- border: 0;
- padding: 1px;
- background: linear-gradient(135deg, red, orange, green, #16cfd0, purple);
- --mask-image: repeating-linear-gradient(135deg, #000 0px, #000 1px, transparent 1px, transparent 4px);
- -webkit-mask-image: var(--mask-image);
- mask-image: var(--mask-image);
- .visualizing-container-content-left {
- width: $lrWidth;
- background: linear-gradient(to right, rgba(0, 0, 0, 0.9), rgba(0, 0, 0, 0.5), rgba(0, 0, 0, 0.01));
- padding-top: 60px;
- .visualizing-container-content-left-flex {
- height: 50%;
- .visualizing-container-content-center {
- height: $cheight;
- background: linear-gradient(to top, rgba(0, 0, 0, 0.9), rgba(0, 0, 0, 0.5), rgba(0, 0, 0, 0.01));
- .visualizing-container-content-center-bottom {
- width: calc(100% - #{$lrWidth * 2});
- left: $lrWidth;
- .visualizing-container-content-center-bottom-flex {
- .visualizing-container-content-right {
- background: linear-gradient(to left, rgba(0, 0, 0, 0.9), rgba(0, 0, 0, 0.5), rgba(0, 0, 0, 0.01));
- .visualizing-container-content-right-flex {
@@ -1,1340 +0,0 @@
- <div class="visualizing-demo2">
- <!-- 顶部 -->
- <div class="big-data-up">
- <SvgIcon name="ele-Timer" class="mr5" />
- <div class="up-right">
- <el-dropdown size="small">
- <span class="el-dropdown-link">
- {{ dropdownActive }}
- <SvgIcon name="ele-ArrowDown" class="el-icon--right" />
- <template #dropdown>
- <el-dropdown-menu>
- <el-dropdown-item v-for="(v, k) in dropdownList" :key="k">{{ v.label }} </el-dropdown-item>
- </el-dropdown-menu>
- </el-dropdown>
- <div class="ml15">
- <SvgIcon name="ele-Bell" class="mr5" />
- <span>消息</span>
- <SvgIcon name="ele-User" class="mr5" />
- <span>个人</span>
- <SvgIcon name="ele-SwitchButton" class="mr5" />
- <span>返回</span>
- <div class="big-data-down">
- <span class="font">多云转晴</span>
- <span class="font">东南风</span>
- <span>良</span>
- <div class="sky-tip">温馨提示:多云转晴,南风转北风风力3级</div>
- <span class="font">°C</span>
- <div>{{ v.v4 }}</div>
- <div>{{ v.v6 }}</div>
- <div class="d-btn-item" v-for="(v, k) in dBtnList" :key="k" :class="{ 'd-btn-active': dBtnActive === k }">
- <SvgIcon name="ele-Money" class="d-btn-item-left" />
- <div class="flex-content" ref="rightChartData1"></div>
- <div class="big-data-down-center-one-content" ref="rightChartData5">
- <div id="3DEarth"></div>
- <div :class="v.topLevelClass" v-for="(v, k) in earth3DBtnList" :key="k">
- <div class="circle" v-for="i in 4" :key="i"></div>
- <div class="text-box">
- <div class="text">{{ v.label }}</div>
- <div class="monitor-wave" :class="{ 'monitor-active': k === chartData4Index }">
- <div class="flex-content-right" ref="rightChartData4"></div>
- <div class="flex-content" ref="rightChartData3"></div>
- <div class="task-item-box task4">
- <div class="task-item-label">巡园</div>
- <div class="task-item-box task5">
- <div class="task-item-value">2</div>
- <div class="task-item-label">采集</div>
- <div ref="rightChartData6" class="progress"></div>
- <div class="flex-content" ref="rightChartData2"></div>
-import 'echarts-gl';
-import { dropdownList, skyList, dBtnList, earth3DBtnList, chartData4List } from './mock/demo2';
-import worldImg from './images/world.jpg';
-import bathymetryImg from './images/bathymetry.jpg';
- dropdownList,
- dropdownActive: '请选择',
- chartData4Index: 0,
- dBtnActive: 0,
- earth3DBtnList,
- // 近30天预警总数
- const initRightChartData1 = () => {
- const myChart = echarts.init(proxy.$refs.rightChartData1);
- radius: [10, 60],
- borderRadius: 5,
- { name: '天气预警', value: 100 },
- { name: '病虫害预警', value: 50 },
- { name: '任务预警', value: 130 },
- { name: '监测设备预警', value: 62 },
- color: '#c0d1f2',
- // 当前设备监测
- const initRightChartData4 = () => {
- const myChart = echarts.init(proxy.$refs.rightChartData4);
- top: 10,
- name: '温度',
- data: [140, 232, 101, 264, 90, 70],
- // 近7天产品追溯扫码统计
- const initRightChartData3 = () => {
- const myChart = echarts.init(proxy.$refs.rightChartData3);
- // 当前任务统计
- const initRightChartData6 = () => {
- const myChart = echarts.init(proxy.$refs.rightChartData6);
- top: 20,
- right: 50,
- bottom: 0,
- left: 80,
- //y轴
- inverse: true,
- data: ['施肥任务完成率', '施药任务完成率', '农事任务完成率'],
- color: '#A7D6F4',
- fontSize: 12,
- name: '标准化',
- barWidth: 10, // 柱子宽度
- position: 'right', // 位置
- distance: 15, // 距离
- formatter: '{c}%', // 这里是数据展示的时候显示的数据
- }, // 柱子上方的数值
- barBorderRadius: [0, 20, 20, 0], // 圆角(左上、右上、右下、左下)
- color: new echarts.graphic.LinearGradient(
- 1,
- 0,
- color: '#51C5FD',
- color: '#005BB1',
- false
- ), // 渐变
- data: [75, 100, 60],
- // 近7天投入品记录
- const initRightChartData2 = () => {
- const myChart = echarts.init(proxy.$refs.rightChartData2);
- // 3DEarth 地图
- const init3DEarth = (globeRadius) => {
- let el = document.getElementById('3DEarth');
- el.style.height = `${proxy.$refs.rightChartData5.offsetHeight}px`;
- const myChart = echarts.init(el);
- globe: {
- baseTexture: worldImg,
- heightTexture: bathymetryImg,
- shading: 'realistic',
- light: {
- ambient: {
- intensity: 0.4,
- main: {
- viewControl: {
- autoRotate: true,
- postEffect: {
- enable: true,
- bloom: {
- globeRadius,
- series: {
- type: 'lines3D',
- coordinateSystem: 'globe',
- blendMode: 'lighter',
- color: 'rgb(50, 50, 150)',
- // 随机模拟攻击线
- let rodamData = function () {
- let longitude = 105.18;
- let longitude2 = Math.random() * 360 - 180;
- let latitude = 37.51;
- let latitude2 = Math.random() * 180 - 90;
- coords: [
- [longitude2, latitude2],
- [longitude, latitude],
- value: (Math.random() * 3000).toFixed(2),
- for (let i = 0; i < 150; i++) {
- option.series.data = option.series.data.concat(rodamData());
- // 监听地球大小变化
- const initAddEventListener3DEarth = () => {
- let w = document.body.clientWidth;
- let globeRadius = 0;
- if (w >= 1920) globeRadius = 100;
- else if (w > 1200 && w < 1920) globeRadius = 70;
- else if (w > 992 && w < 1200) globeRadius = 60;
- else if (w > 768 && w < 992) globeRadius = 40;
- else if (w < 768) globeRadius = 20;
- init3DEarth(globeRadius);
- initAddEventListener3DEarth();
- await initRightChartData1();
- await initRightChartData4();
- await initRightChartData3();
- await initRightChartData2();
- await initRightChartData6();
-.visualizing-demo2 {
- background: url(https://gitee.com/lyt-top/vue-next-admin-images/raw/master/visualizing/demo2.jpg);
- background-size: 100% 100%;
- .big-data-up {
- height: 70px;
- color: #43bdf0;
- font-style: italic;
- background-image: -webkit-linear-gradient(left, #43bdf0, #c0d1f2 25%, #43bdf0 50%, #c0d1f2 75%, #43bdf0);
- animation: masked-animation 4s infinite linear;
- -webkit-box-reflect: below -2px -webkit-gradient(linear, left top, left bottom, from(rgba(0, 0, 0, 0)), to(rgba(255, 255, 255, 0.1)));
- border-image: linear-gradient(to right, rgba(0, 0, 0, 0.1), #43bdf0) 1 10;
- .up-right {
- .ml15:hover {
- ::v-deep(.el-dropdown) {
- font-size: 13px !important;
- .big-data-down {
- background: rgba(22, 34, 58, 0.4);
- color: #c0d1f2;
- background: unset !important;
- border-radius: unset !important;
- padding: unset !important;
- .sky-tip {
- background: rgba(0, 0, 0, 0.05);
- &:first-child:hover {
- background: unset;
- border: 1px solid #c0d1f2;
- padding: 0 4px;
- .d-btn-active {
- border: 1px solid #43bdf0;
- .fixed-top,
- .fixed-right,
- .fixed-bottom,
- .fixed-left {
- width: 100px;
- height: 100px;
- .circle {
- background: rgba(0, 0, 0, 0.01);
- z-index: 10;
- .text-box {
- z-index: 11;
- font-size: 28px;
- .fixed-top {
- left: 20px;
- top: 20px;
- .fixed-right {
- right: 20px;
- .fixed-bottom {
- bottom: 20px;
- .circle:nth-of-type(1) {
- height: 95px;
- animation: turnAround 6s infinite linear;
- box-shadow: 0 0 1px 0 #869fe4, inset 0 0 10px 0 #869fe4;
- .circle:nth-of-type(2) {
- width: 95px;
- animation: turnAround 10s infinite linear;
- box-shadow: 0 0 1px 0 #3397f2, inset 0 0 10px 0 #3397f2;
- .circle:nth-of-type(3) {
- width: 110px;
- animation: turnAround 5s infinite linear;
- box-shadow: 0 0 1px 0 #3eaadc, inset 0 0 10px 0 #3eaadc;
- .circle:nth-of-type(4) {
- height: 110px;
- animation: turnAround 15s infinite linear;
- box-shadow: 0 0 1px 0 #09f, inset 0 0 10px 0 #09f;
- @keyframes turnAround {
- transform: rotate(360deg);
- background-color: #43bdf0;
- color: #4eb8ff;
- .task4 {
- background: #b4825a;
- .task5 {
- background: #74568f;
-export const echartsMapList: Array<object> = [
- { name: '深圳市人民政府', value: '100' },
- { name: '莲花山公园', value: '100' },
- { name: '世界之窗', value: '100' },
- { name: '华侨城欢乐谷', value: '100' },
- { name: '宝安区西乡', value: '100' },
-export const echartsMapData: object = {
- 深圳市人民政府: [114.064524, 22.549225],
- 莲花山公园: [114.0658, 22.560072],
- 世界之窗: [113.979419, 22.540579],
- 华侨城欢乐谷: [113.986066, 22.548056],
- 宝安区西乡: [113.869053, 22.581714],
-// 地图图片显示
-export const echartsMapImgs: Array<object> = [
- url: 'https://img1.baidu.com/it/u=4244861097,3561366422&fm=11&fmt=auto&gp=0.jpg',
- name: '深圳市人民政府',
- add: '深圳市福田区福中三路市民中心C区',
- dec: '深圳市人民政府是根据《中华人民共和国地方各级人民代表大会和地方各级人民政府组织法》设立的,是深圳市人民代表大会的执行机关,是深圳市的国家行政机关。',
- url: 'https://img1.baidu.com/it/u=3793608028,4006842751&fm=26&fmt=auto&gp=0.jpg',
- name: '莲花山公园',
- add: '广东省深圳市福田区莲花街道莲花北社区红荔路6030号',
- dec: '莲花山公园筹建于1992年10月10日 ,1997年6月23日正式对外局部开放。',
- url: 'https://img0.baidu.com/it/u=1406340112,1927292660&fm=26&fmt=auto&gp=0.jpg',
- name: '世界之窗',
- add: '深圳市南山区深南大道9037号',
- dec: '这里,世界首座实景拍摄悬空式球幕影院“飞跃美利坚””,为游客提供集休闲放松于一体的都市时尚生活空间。',
- url: 'https://img0.baidu.com/it/u=3042342330,902556630&fm=26&fmt=auto&gp=0.jpg',
- name: '华侨城欢乐谷',
- add: '广东省深圳市南山区沙河街道星河街社区侨城西街1号',
- dec: '深圳欢乐谷注重满足人们参与、体验的新型诱游需求,营造出自然、清新、活泼、惊奇、热烈、刺激的休闲旅游氛围。',
- url: 'https://img2.baidu.com/it/u=1075072079,1229283519&fm=11&fmt=auto&gp=0.jpg',
- name: '宝安区西乡',
- add: '西乡街道下辖25个社区',
- dec: '西乡街道,隶属于广东省深圳市宝安区,位于宝安区西南部,东接石岩街道,南接新安街道,西至珠江口岸边,北接航城街道。',
@@ -1,131 +0,0 @@
-// 顶部下来菜单
-export const dropdownList: Array<object> = [
- label: '广东省农业农村厅',
- label: '广西省农业农村厅',
- label: '四川省农业农村厅',
- label: '湖北省农业农村厅',
- label: '福建省农业农村厅',
- label: '山东省农业农村厅',
- label: '江西省农业农村厅',
-// sky 天气
-export const skyList: Array<object> = [
- v4: '湿度',
- v5: '降水概率',
- v6: '风向',
- v4: '80%',
- v6: '东南风',
- v1: '后天',
-// 当前设置状态
-export const dBtnList: Array<object> = [
- v1: '地块A-灌溉',
- v1: '地块B-收割',
- v3: '360天',
- v4: '设备预警',
-// 当前设备监测
-export const chartData4List: Array<object> = [
- label: '张力',
- label: '气压',
-// 3DEarth 地图周围按钮组
-export const earth3DBtnList: Array<object> = [
- topLevelClass: 'fixed-top',
- icon: 'ele-MagicStick',
- label: '环境监测',
- type: 0,
- topLevelClass: 'fixed-right',
- icon: 'ele-MoonNight',
- label: '精准管理',
- type: 1,
- topLevelClass: 'fixed-bottom',
- icon: 'ele-TrendCharts',
- label: '数据报表',
- type: 2,
- topLevelClass: 'fixed-left',
- icon: 'ele-Van',
- label: '产品追溯',
- type: 3,