当前位置: 首页 > article >正文

数据权限的设计与实现系列12——前端筛选器组件Everright-filter集成功能完善3

筛选条件数据类型完善

数值类

筛选器组件给了一个数值类类操作的范例,如下:

             Number: [
                {
                  label: '等于',
                  en_label: 'Equal',
                  value: 'equal',
                  style: 'noop'
                },
                {
                  label: '不等于',
                  en_label: 'Not equal',
                  value: 'not_equal',
                  style: 'noop'
                },
                {
                  label: '大于',
                  en_label: 'Greater than',
                  value: 'greater_than',
                  style: 'noop'
                },
                {
                  label: '大于等于',
                  en_label: 'Greater than or equal to',
                  value: 'greater_than_equal',
                  style: 'noop'
                },
                {
                  label: '小于',
                  en_label: 'Less than',
                  value: 'less_than',
                  style: 'noop'
                },
                {
                  label: '小于等于',
                  en_label: 'Less than or equal to',
                  value: 'less_than_equal',
                  style: 'noop'
                },
                {
                  label: '区间',
                  en_label: 'Between',
                  value: 'between',
                  style: 'range'
                },
                {
                  label: '为空',
                  en_label: 'Empty',
                  value: 'empty',
                  style: 'none'
                },
                {
                  label: '不为空',
                  en_label: 'Not empty',
                  value: 'not_empty',
                  style: 'none'
                }
              ]
            }

平台进行以下调整:

  1. 区间可以可以等价转换一个条件组,内含两个条件,大于等于最小值,小于等于最大值,组内逻辑关系为且。而且可以只设置其中一部分,更灵活,因此去除区间。
  2. 对操作符的编码做相应调整。

调整完成后如下:

             Number: [
                {
                  label: '等于',
                  value: 'EQ',
                  style: 'noop'
                },
                {
                  label: '大于',
                  value: 'GT',
                  style: 'noop'
                },
                {
                  label: '大于等于',
                  value: 'GE',
                  style: 'noop'
                },
                {
                  label: '小于',
                  value: 'LT',
                  style: 'noop'
                },
                {
                  label: '小于等于',
                  value: 'LE',
                  style: 'noop'
                },

                {
                  label: '不等于',
                  value: 'NE',
                  style: 'noop'
                },
                {
                  label: '为空',
                  value: 'NL',
                  style: 'none'
                },
                {
                  label: '不为空',
                  value: 'NN',
                  style: 'none'
                }
              ]

获取规则条件时,需要将数值类的操作符指定为Number,做以下调整:


    /**
     * 获取完整属性列表,转换为过滤器需要的格式
     * 只取数据库存储的属性
     *
     * @param entityModelId 实体模型id
     * @return {@link ResponseEntity}<{@link Result}>
     */
    @GetMapping("/{entityModelId}/getFullPropertyListForFilter")
    @SystemLog(value = "实体模型属性-完整列表,转换为过滤器格式")
    @PreAuthorize("hasPermission(null,'entityconfig:entityModelProperty:query')")
    public ResponseEntity<Result> getFullPropertyListForFilter(@PathVariable String entityModelId) {
        // 构造查询条件

        List<EntityModelProperty> list = entityModelPropertyService.getFullPropertyByEntityModelId(entityModelId);
        // 转换vo
        List<EntityModelPropertyForFilterVO> entityModelPropertyVOList = new ArrayList<>();
        for (int i = 0; i < list.size(); i++) {
            // 获取属性定义
            EntityModelProperty entityModelProperty = list.get(i);
            // 非库表存储属性直接跳过
            if (entityModelProperty.getDatabaseStoreFlag().equals(YesOrNoEnum.NO.name())) {
                continue;
            }

            // 获取平台展现属性的控件类型
            String dataType = entityModelProperty.getDataType();
            String widgetType = entityModelProperty.getWidgetType();
            String[] numberDataType = {"INTEGER", "LONG", "DOUBLE", "DECIMAL"};
            String renderType = "TEXT";
            String operatorKey = "Text";

            switch (widgetType) {
                case "TEXT":
                    if (ArrayUtils.contains(numberDataType, dataType)) {
                        renderType = "NUMBER";
                        operatorKey = "Number";
                    } else {
                        renderType = "TEXT";
                        operatorKey = "Text";
                    }
                    break;
                case "TEXTAREA":
                case "RICH_TEXT":
                    renderType = "TEXT";
                    break;
                case "DATETIME":
                    renderType = "DATETIME";
                    break;
                case "TIME":
                    renderType = "TIME";
                    break;
                case "DROP_DOWN_LIST":
                case "RADIO_BUTTON_GROUP":
                    renderType = "SELECT";
                    break;
                default:
                    renderType = "";
            }
            if (StringUtils.isNotBlank(renderType)) {
                EntityModelPropertyForFilterVO vo = new EntityModelPropertyForFilterVO();
                vo.setLabel(entityModelProperty.getName());
                vo.setValue(entityModelProperty.getCode());
                vo.setRenderType(renderType);
                vo.setOperatorKey(operatorKey);
                entityModelPropertyVOList.add(vo);
            }

        }
        return ResultUtil.success(entityModelPropertyVOList);
    }

后端将规则转换成SQL片段的功能方法在文本类已经相应处理过了,测试如下:

符合预期。

日期时间类

日期时间类原来计划是需要处理的,做的过程中意识到,实际业务需求不会存在按照日期和时间来配置数据权限规则的。
因此在返回筛选条件时,将日期时间类的属性也去除掉。

数据字典

接下来重点攻坚数据字典。
从数据权限规则配置角度,同样我们不使用多选,有需要时转换成多个条件就好了。
官方文档中这块说得很少,除了上面常规属性外,增加了两个属性,multiple是否可以多选,multipleLimit,限制多选的最大数量,如下图:

但是下拉的数据源哪来的,如何赋值呢?
仔细观察上面例子,原先的操作符里放了Gender,看上去像是数据字典类型的编码。
然后在API Conditions部分,就放了一个select的示例,啥也没说……

只能动手摸索了。

还是拿先前的功能验证页面进行快速验证,做如下调整:
为筛选器组件绑定方法,如下:

然后在原有筛选条件的基础上,增加一个性别的条件,并在操作属性中同步新增一个Gender对象,如下:

最后,就是下拉列表的数据的如何构建了,尝试如下:

const getConditions = async (params) => {
  return new Promise((resolve, reject) => {
    resolve({
      data: {
        conditions: {
          gender: [
            {
              label: '男',
              en_label: 'male',
              value: 'male'
            },
            {
              label: '女',
              en_label: 'female',
              value: 'female'
            }
          ]
        }
      }
    })
  })
}

运行后无法加载下拉数据,效果如下:

控制台报错:

尝试将conditions中的gender更改为大写Gender,错误依旧……

关键在于,下拉列表的数据源,是如何跟筛选条件关联到一起的,在option对象不增加属性的情况下,要不然是复用了操作operatorKey,要不然是复用值value,后者的可能性更大,实际测试,两条路都不通,这就难办了……

再翻了一遍官方文档和demo,没有找到例子,百度和ai搜索也没有结果,还得继续摸索。

然后看到getConditions 方法是带参数的, async (params),这里面的params推测是数据字典类型的编码gender(对应着筛选条件的value值),用console.log验证了下,果然是这样,然后改写了下处理:

const getConditions = async (params) => {
  const { property } = params
  console.log(property)
  if (property === 'gender') {
    // 当 property 为 'gender' 时,返回相应的条件选项
    return new Promise((resolve, reject) => {
      resolve({
        data: {
          conditions: [
            { label: '男', value: 'male' },
            { label: '女', value: 'female' }
          ]
        }
      })
    })
  }
}

结果还是不行。
精简数据层次结构,去除了conditions节点,直接让data包裹数据,如下:

const getConditions = async (params) => {
  const { property } = params

  if (property === 'gender') {
    console.log(property)
    // 当 property 为 'gender' 时,返回相应的条件选项
    return new Promise((resolve, reject) => {
      resolve({
        data: [
          { label: '男', value: 'male' },
          { label: '女', value: 'female' }
        ]
      })
    })
  }
}

下拉数据源终于实现了,如下:


demo验证通过了,开始做正式功能。
这里有个问题需要解决,筛选器组件复用了条件对象的value属性,即要求数据字典类型的编码必须跟条件的编码完全一致,这样会存在限制,因为实际条件的属性编码跟数据字典类型编码通常是不一样的。
该参数是内部传递的,无法扩展,因此只能采用变通的方式,我们在获取到实体属性定义的时候,将属性编码和数据字典类型对应关系先缓存下来,然后依据缓存的对应关系找到数据字典类型编码,然后再调用后端服务获取字典项加载。

首先,前后端交互的视图对象类型中,新增数据字典类型属性,如下:

后端查询实体模型属性的时候,如果该属性为数据字典,则写入字典类型编码,同时操作定义统一为DataDictionary,如下:

前端在调用实体属性列表时缓存对应关系,如下:

前端添加筛选条件为数据字典时,触发数据字典项查询服务,如下:

 async getConditions(params) {
      // 根据缓存的对应关系,获取到属性对应的数据字典类型编码
      const dictionaryType = this.propertyDictionaryTypeMapping[params.property]
      return new Promise((resolve, reject) => {
        this.$api.system.dictionaryType.getItem(dictionaryType).then((res) => {
          if (res.data.length > 0) {
            res.data = res.data.map((item) => {
              // 后端范围的value是id,code才是编码,因此需要转换
              return { label: item.label, value: item.code }
            })
          }
          resolve(res)
        })
      })
    }

还有就是给数据字典类型新增一个固定的操作集合,只有等于或不等于即可,如下:


最后,运行效果如下:

点击生成规则以及转换SQL片段,符合预期,如下:

开源平台资料

平台名称:一二三开发平台
简介: 企业级通用开发平台
设计资料:[csdn专栏]
开源地址:[Gitee]
开源协议:MIT
如果您在阅读本文时获得了帮助或受到了启发,希望您能够喜欢并收藏这篇文章,为它点赞~
请在评论区与我分享您的想法和心得,一起交流学习,不断进步,遇见更加优秀的自己!


http://www.kler.cn/a/354027.html

相关文章:

  • SpringAI从入门到熟练
  • WebSocket封装
  • pyinstaller打包exe可执行文件
  • 工作中常用Vim的命令
  • android app构建时排除指定类
  • 四年匠心磨砺,快手系统软件技术创新与领域演进之路
  • 优化SpringBoot接口:异步处理提升系统吞吐量策略
  • SQL语句查询
  • 【IC设计】复旦微行业分析
  • 为什么你总碰到渣男?伯克森悖论
  • 博客搭建之路:hexo使用next主题渲染流程图
  • 技术总结(七)
  • 解决ultralytics中的YOLOv8在执行task.py文件添加模块操作出现的KeyError报错
  • Linux-lvs
  • 芒果YOLOv10改进136:注意力机制MLLA|即插即用:融合Mamba设计精髓的线性注意力视觉Transformer
  • Ubuntu(Linux)tcpdump使用方法详解
  • 金融信用评分卡建模项目:AI辅助
  • mysql指令笔记(基本)
  • C#/WinForm 自定义控件绘制章鱼
  • 【2022工业3D异常检测文献】Patch+FPFH: 结合3D手工点云描述符和颜色特征的异常检测方法
  • xlsx xlsx-style-vite 实现前端根据element 表格导出excel且定制化样式 背景 列宽等
  • 【网络安全】-vulnhub靶场-noob
  • 顺序表的查找
  • 如何将闲置平板变为电脑显示器?GameViewer远程助你低成本实现0门槛副屏串流!
  • 基于Redis实现的延迟队列
  • MATLAB基础应用精讲-【数模应用】HLM模型