vue2实现公式规则编辑校验弹窗功能

news2025/1/6 19:17:06

文章目录

    • 需求描述
    • 技术栈
    • 最终效果演示
    • 功能实现
        • 逻辑拆分
        • 代码目录结构
        • 实现思路
          • 光标实现
          • 底部单个符号或字段结构设计
          • 监听键盘事件&处理光标
          • 公式规则校验
      • 总结

需求描述

    需要一个弹窗,弹窗内部需要能够进行公式规则的配置并进行公式规则合法性校验。

技术栈

  • vue2
  • element-ui

最终效果演示

公式规则功能效果演示

功能实现

逻辑拆分

    我将弹窗大致拆分成了三部分,具体如下:

代码目录结构

实现思路

光标实现

使用的是div块级元素+css3动画实现,核心代码如下:

// template
<div v-if="item.show" class="expression-blink-cursor" :key="item.id"></div>

// style
.expression-blink-cursor {
  width: 1px;
  height: 18px;
  border-left: 1px solid;
  margin: 2px;
  animation:cursorImg 1s infinite steps(1, start);
  @keyframes cursorImg {
    0%, 100% {
      opacity: 0;
    }
    50% {
      opacity: 1;
    }
  }
}
底部单个符号或字段结构设计

实现公式规则的编辑功能主要是通过操作数组实现的,所以对于弹窗底部的符号或字段设计为对象数组,便于后续数组操作。部分代码如下:

// 子组件template
<template>
  <div id="singleSymbol">
    <template v-for="item in symbolBtnList">
      <template v-if="item.code === 'formFields'">
        <el-cascader class="custom-cascader-class" v-model="selectedFormFieldsValue" :options="formFieldsOptions" :key="item.id" :placeholder="item.label" @change="selectFormFieldsChange">
          <template slot-scope="{ node, data }">
            <span>{{ data.label }}</span>
            <span v-if="!node.isLeaf"> ({{ data.children.length }}) </span>
          </template>
        </el-cascader>
      </template>
      <template v-else>
        <el-button class="symbol-btn" :key="item.code" @click="singleSymbolClick(item)">
          {{item.label}}
        </el-button>
      </template>
    </template>
  </div>
</template>

// js
import {nanoid} from 'nanoid'
export const singleSymbolBtnList = [
  {
    label: '表单字段',
    code: 'formFields',
    templateName: 'formFields',
    type: 'customTemplate',
    id: nanoid()
  },
  {
    label: '+',
    code: 'plusSign',
    type: 'bottomSingleSign',
    id: nanoid()
  },
  ...
]
export const formFieldsOptions = [
  {
    value: 'testForm1',
    label: '测试表单1',
    children: [
      {
        value: 'formFields1',
        uuid: '',
        label: '表单字段1'
      },
      {
        value: 'formFields2',
        uuid: '',
        label: '表单字段2'
      },
      {
        value: 'formFields3',
        uuid: '',
        label: '表单字段3'
      },
      {
        value: 'formFields4',
        uuid: '',
        label: '表单字段4'
      }
    ]
  },
  {
    value: 'testForm2',
    label: '测试表单2',
    children: [
      {
        value: 'formFields1',
        uuid: '',
        label: '表单字段1'
      },
      {
        value: 'formFields2',
        uuid: '',
        label: '表单字段2'
      },
      {
        value: 'formFields3',
        uuid: '',
        label: '表单字段3'
      },
      {
        value: 'formFields4',
        uuid: '',
        label: '表单字段4'
      }
    ]
  }
]

// 父组件template
<template>
  <div id="formulaRulesMain" class="formula-rules-main">
    <!-- 公式规则配置区域 -->
    <template v-for="(item, index) in formulaRulesArray">
      <template v-if="item.type === 'customTemplate'">
        <template v-if="item.templateName === 'blinkCursor'">
          <div v-if="item.show" class="expression-blink-cursor" :key="item.id"></div>
        </template>
        <template v-if="item.templateName === 'staticValue'">
          <el-input type="text" v-focus class="custom-input" :key="item.id" v-model="item.bindValue" @focus="inputFocusOrBlur('focus')" @blur="inputFocusOrBlur('blur')"></el-input>
        </template>
        <template v-if="item.templateName === 'booleanValue'">
          <div class="custom-boolean-value" :key="item.id">{{item.label}}</div>
        </template>
        <template v-if="item.templateName === 'formFields'">
          <div class="custom-form-fields" :key="item.id">{{item.label}}</div>
        </template>
        <template v-if="item.templateName === 'emptyDiv'">
          <div class="custom-form-empty-div" :key="item.id" @click="emptyDivClick(index)"></div>
        </template>
      </template>
      <template v-else>
        <template v-if="item.type === 'bottomSingleSign'">
          <div :key="item.id">
            <span class="expression-item">{{item.label}}</span>
          </div>
        </template>
      </template>
    </template>
  </div>
</template>
监听键盘事件&处理光标

    在mounted挂载键盘监听事件,并对不同按键操作做出不同逻辑处理(对数组进行操作),核心代码如下:

  mounted () {
    this.keyDown()
  },
  methods: {
    // 监听键盘
    keyDown () {
      window.onkeyup = (e) => {
        // 事件对象兼容
        // console.log('查看对象兼容', e, e.keyCode)
        let e1 = e || event || window.event
        // 键盘按键判断:左箭头-37;上箭头-38;右箭头-39;下箭头-40;删除键-8
        // 左
        if (e1 && e1.keyCode === 37) {
          // 按下左箭头
          this.handleKeydownEvent('left')
        } else if (e1 && e1.keyCode === 39) {
          // 按下右箭头
          this.handleKeydownEvent('right')
        } else if (e1 && e1.keyCode === 8) {
          // 按下删除键
          this.handleKeydownEvent('delete')
        }
      }
    },
    // 处理键盘事件
    handleKeydownEvent (type) {
      let arr = this.formulaRulesArray
      if (type === 'left') {
        if (this.cursorIndex !== 0) arr[this.cursorIndex] = arr.splice(this.cursorIndex - 2, 1, arr[this.cursorIndex])[0];
      } else if (type === 'right') {
        if (this.cursorIndex !== arr.length - 1) arr[this.cursorIndex] = arr.splice(this.cursorIndex + 2, 1, arr[this.cursorIndex])[0];
      } else if (type === 'delete') {
        if (this.cursorIndex !== 0) arr.splice(this.cursorIndex - 2, 2)
      }
    },
  }
公式规则校验

    通过上述步骤 基本的编辑功能已经实现,只差一步校验就完成所需功能了。
    最开始考虑的是通过正则去判断公式规则的合法性,但是整不出来😊 所以换了另外一种思路。采用关键字替换然后使用eval函数执行调用自定义函数看是否能够成功执行。

  • 首先可以校验一下左右括号个数是否匹配 比较简单
  • 检验完括号后 如果匹配 调用eval函数执行公式规则字符串。
  • js函数能够支持多个传参 省参执行,js函数会通过实参(arguments)对象接收,无论调用函数时传递了多少个参数,都会被arguments接收,利用这一点就可以替换完关键字后直接调用自定义函数通过js引擎去帮我们“校验”公式规则的合法性😄。
    核心代码如下:
methods: {
    // 处理公式规则结构类型
    handleFormulaArray (formulaArr) {
      let tempArr = Array.from(formulaArr, ({label}) => label)
      let finalArr = []
      formulaArr.forEach(item => {
        let each = item.label
        switch (item.label) {
          case 'SUM':
            each = 'this.sum'
            break;
          case 'SUB':
            each = 'this.sub'
            break
          case '静态值':
            each = item.bindValue
            break
          case 'TRUE':
            each = true
            break
          case 'FALSE':
            each = false
            break
          case 'IF':
            each = 'this.judgeIf'
            break
          case 'INT1':
            each = 'this.int1'
            break
          case 'INT2':
            each = 'this.int2'
            break
          case 'INT3':
            each = 'this.int3'
            break
        }
        if (item.code === 'formFields') {
          // TODO 通过uuid获取到表单对应字段的值,item有完整数据(uuid...)
          each = 5
        }
        finalArr.push(each)
      })
      console.log('查看处理后的数组: ', finalArr, finalArr.join(''))
      try {
        if (finalArr.length === 0) {
          this.$message({
            message: '校验失败,公式规则不可为空!',
            type: 'warning'
          });
          return
        }

        this.showResultVisible = true
        this.resltFormulaStr = finalArr.join('')
        const str = finalArr.join('')
        if (!eval(str)) throw new Error()
        this.showResultVisible = true
        this.resltFormulaStr = finalArr.join('')
        this.resultFormulaValue = eval(str)
        // TODO 预留存储公式规则逻辑 包括公式规则完整信息formulaArr及转化后的字符串str、校验成功后进行字段赋值操作及关闭弹窗
        // this.dialogVisible = false
      } catch (error) {
        this.$message({
          message: '校验失败,请检查公式规则!',
          type: 'warning'
        });
        this.resultFormulaValue = null
      }
    }
}

// mixin
export default {
  methods: {
    // -------------------------------------------  数学函数  -----------------------------------------------
    // 求和
    sum () {
      try {
        if (!arguments[0] || arguments.length < 2) throw new Error()
        let result = 0
        for (let i = 0; i < arguments.length; i++) {
          result = result + arguments[i]
        }
        return result
      } catch (error) {
        this.$message({
          message: '校验失败,请检查SUM公式规则!',
          type: 'warning'
        });
      }
    },
    // 相减
    sub () {
      try {
        if (!arguments[0] || arguments.length < 2) throw new Error()
        let result = arguments[0]
        for (let i = 1; i < arguments.length; i++) {
          result = result - arguments[i]
        }
        return result
      } catch (error) {
        this.$message({
          message: '校验失败,请检查SUB公式规则!',
          type: 'warning'
        });
      }
    },
    // if判断函数
    judgeIf (judgeCode, trueCode, falseCode) {
      try {
        if (arguments.length !== 3) throw new Error()
        return judgeCode ? trueCode : falseCode
      } catch (error) {
        this.$message({
          message: '校验失败,请检查IF公式规则!',
          type: 'warning'
        });
      }
    },
    // 向下取整
    int1 () {
      try {
        if (!arguments[0]) throw new Error()
        return parseInt(arguments[0])
      } catch (error) {
        this.$message({
          message: '校验失败,请检查向下取整函数INT1公式规则!',
          type: 'warning'
        });
      }
    },
    // 向上取整
    int2 () {
      try {
        if (!arguments[0]) throw new Error()
        return parseInt(arguments[0]) + 1
      } catch (error) {
        this.$message({
          message: '校验失败,请检查向上取整函数INT2公式规则!',
          type: 'warning'
        });
      }
    },
    // 四舍五入取整
    int3 () {
      try {
        if (!arguments[0]) throw new Error()
        return Math.round(arguments[0])
      } catch (error) {
        this.$message({
          message: '校验失败,请检查四舍五入取整函数INT3公式规则!',
          type: 'warning'
        });
      }
    }
  }
}

总结

    遇到较难的问题可以变通一下去考虑解决方案(正则太难了😭), 然后一步步去验证,此需求实现代码是我的一个小demo,不完善的地方还有很多,望大佬们能够指正,当然,基本功能是完全OK的。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/693071.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Thymeleaf的常用语法

&#x1f31f; Thymeleaf的常用语法 Thymeleaf是一个Java模板引擎&#xff0c;用于处理HTML、XML、JavaScript、CSS等文件。它可以与Spring框架无缝集成&#xff0c;为Web应用程序提供优雅的模板解决方案。本文将介绍Thymeleaf的常用语法&#xff0c;包括th属性、表达式、内置…

Spring Boot 中的 @RestController 注解,如何使用

Spring Boot 中的 RestController 注解 在 Spring Boot 中&#xff0c;我们经常需要编写 RESTful Web 服务&#xff0c;以便于客户端与服务器之间的通信。为了简化 RESTful Web 服务的开发&#xff0c;Spring Boot 提供了 RestController 注解&#xff0c;它可以让我们更方便地…

Jmeter实现参数加密

目录 一、使用__digest自带函数 以md5加密算法演示使用方法 二、在BeanShell 中使用JAVA代码实现算法加密 规避BUG的方法 JMeter有两种方法可以实现算法加密 一、使用__digest自带函数 参数说明&#xff1a; Digest algorithm&#xff1a;算法摘要&#xff0c;可输入值&a…

如何避免在处理数据时出现错误?

介绍 当业务运营管道中发生大量事情时&#xff0c;数据输入任务的优先级往往较低。但是&#xff0c;数据输入被认为是所有行业部门中经常发生的最重要的任务之一。从销售数据到客户分析&#xff0c;从财务数据到库存管理&#xff0c;每项工作都依赖于数据输入&#xff0c;这意…

前端vue入门(纯代码)16

【18.如何在Vue中配置代理服务器】 vue脚手架配置代理总结&#xff1a;修改了vue.config.js文件后必须重启项目【npm run serve】 方法一 ​ 在vue.config.js中添加如下配置&#xff1a; devServer:{proxy:"http://localhost:5000" }说明&#xff1a; 优点&#…

家政上门预约小程序;

家政上门预约小程序开发是一款本地生活类服务上门预约系统&#xff0c;覆盖家政保洁、保姆月嫂、上门维修、管道疏通、上门安装等各种到家服务。可以帮助创业者在不需要相关技术人员及大量资金投入的情况下&#xff0c;就能够轻松搭建并运营一个上门家政服务平台。 那么开发一…

【零基础入门学习Python---Python错误处理和异常保姆级教程】

&#x1f680; Python &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

【图像处理OpenCV(C++版)】——5.3 图像平滑之均值平滑(滤波)

前言&#xff1a; &#x1f60a;&#x1f60a;&#x1f60a;欢迎来到本博客&#x1f60a;&#x1f60a;&#x1f60a; &#x1f31f;&#x1f31f;&#x1f31f; 本专栏主要结合OpenCV和C来实现一些基本的图像处理算法并详细解释各参数含义&#xff0c;适用于平时学习、工作快…

Flutter学习四:Flutter开发基础(一)Widget

目录 0 引言 1 Widget 简介 1.1 Widget 概念 1.2 Widget 接口 1.3 Flutter中的四棵树 1.4 StatelessWidget 1.4.1 简介 1.4.2 Context上下文 1.5 StatefulWidget 1.6 State 1.6.1 简介 1.6.2 State生命周期 1.7 在 widget 树中获取State对象 1.7.1 通过Context…

详解如何使用nvm管理Node.js多版本

目录 NVM进行NodeJS多版本管理 背景 安装步骤 1. 下载nvm安装包 2. 安装nvm 使用步骤 下载nodejs 切换版本nodejs ​编辑 常用命令 NVM进行NodeJS多版本管理 背景 有的时候开发环境需要多个NodeJS的版本&#xff0c;这个时候就可以用NVM进行管理。 安装步骤 1. 下载n…

9 从0开始学PyTorch | 过拟合欠拟合、训练集验证集、关闭自动求导

这一小节在开始搞神经网络之前&#xff0c;我们先熟悉几个概念&#xff0c;主要还是把模型训练的流程打通。 过拟合和欠拟合 我们在日常的工作中&#xff0c;训练好的模型往往是要去评价它的准确率的&#xff0c;通过此来判断我们的模型是否符合我的要求。 几个可能的方案是&…

国外学位论文去哪里查找下载

查找下载国外博士论文最合适的文献数据库就是ProQuest学位论文全文数据库。 ProQuest学位论文全文数据库覆盖了大部分北美地区高等院校以及世界其他地区数千个高等院校每年获得通过的博硕士论文。是将ProQuest公司PQDD文摘库&#xff08;现名PQDT&#xff09;中适合中国科研人…

Windows 11 22H2 中文版、英文版 (x64、ARM64) 下载 (updated Jun 2023)

Windows 11 绕过 TPM 方法总结&#xff0c;通用免 TPM 镜像下载 (2023 年 6 月更新) 在虚拟机、Mac 电脑和 TPM 不符合要求的旧电脑上安装 Windows 11 的通用方法总结 请访问原文链接&#xff1a;https://sysin.org/blog/windows-11-no-tpm/&#xff0c;查看最新版。原创作品…

nuxt3 多级动态路由

需求&#xff1a; 写法&#xff1a; 对应 文件目录 pages\product\[class]\[brand]\[SPU].vue pages/ --| product/ ----| [class] ------| [brand] --------| [SPU].vue script 内跳转方法 const router useRouter() const nuxtApp useNuxtApp()const jumpSPU () >…

caffeine和google-guava cache缓存使用详解和源码介绍

google-guava cache 1.pom引入其依赖 <dependency><groupId>com.google.guava</groupId><artifactId>guava</artifactId><version>20.0</version></dependency> 2.具体使用 com.google.common.cache.LoadingCache<Strin…

【selenium】问题记录

1、驱动和浏览器版本不一致 报错&#xff1a;selenium.common.exceptions.SessionNotCreatedException: Message: session not created: This version of ChromeDriver only supports Chrome version 106 问题原因&#xff1a; chrome版本114&#xff0c;Chromedriver版本106 …

机器学习之深度神经网络

目录 卷积神经网络与全连接神经网络 前向后向传播推导 通用手写体识别模型 人脸识别模型 电影评论情感分析模型 卷积神经网络与全连接神经网络 卷积神经网络&#xff08;Convolutional Neural Network&#xff0c;CNN&#xff09;和全连接神经网络&#xff08;Fully Conn…

Django学习笔记-用户名密码登录

笔记内容转载自 AcWing 的 Django 框架课讲义&#xff0c;课程链接&#xff1a;AcWing Django 框架课。 CONTENTS 1. 扩充Django数据库2. 实现获取用户信息3. 渲染登录与注册界面4. 实现登录与登出功能5. 实现注册功能6. 修改获取用户信息 1. 扩充Django数据库 首先我们先在 s…

JavaWeb学习路线(7)——文件上传

一、概念 &#xff08;一&#xff09;文件上传概念&#xff1a; 指将本地的图片、视频、音频等文件上传到服务器&#xff0c;供其他用户浏览或下载的过程。 &#xff08;二&#xff09;前端文件上传三元素 method“post”&#xff08;form&#xff09;enctype“multipart/for…

四、Bean 的作用域,Bean 的自动装配以及通过注解实现 Bean 的自动装配

文章目录 一、Bean 的作用域二、Bean 的自动装配三、通过注解实现 Bean 的自动装配 一、Bean 的作用域 Spring 官网 Bean 的作用域讲解 单例(Singleton)作用域&#xff1a;在这种作用域下&#xff0c;容器只会创建一个Bean实例对象&#xff0c;无论该Bean被注入到多少个其它B…