动态过滤引用字段:关联数据与选择联动
时间:2025-09-10 15:47:29 103浏览 收藏
本文深入解析了ServiceNow中动态过滤引用字段的实现方案,旨在提升数据录入的准确性和用户体验。针对多表关联场景下,如用户与商品关联的新增表单,当用户选择特定用户后,商品选择列表应仅显示未与该用户关联的商品。文章详细阐述了如何通过配置高级引用限定符,并结合服务器端脚本(Script Include),实现基于关联数据和另一引用字段的选择性过滤。文中提供了具体的配置步骤、脚本代码示例及注意事项,强调了表名、字段名准确性的重要性,以及性能优化策略。此方案有效解决了传统数据录入中可能出现的错误和重复关联问题,是ServiceNow平台开发中处理关联数据过滤的实用技巧。
1. 业务场景与挑战
在企业应用开发中,经常需要处理多表关联的数据,并要求在用户输入数据时提供智能化的选择列表。一个典型的场景是:存在用户(Users)和商品(Goods)两种主数据,以及一个记录用户与商品关联关系的链接表(LinkTbl)。现在,我们需要在一个新的表单(NewLinkTbl)中创建新的用户与商品关联。当用户首先选择一个特定的“用户”后,其对应的“商品”选择列表应只显示那些尚未与该用户建立关联的商品。
具体到数据模型,我们有以下四张表:
- UsersTbl: 存储用户信息(UserID, UserName, UserEmail等)。
- GoodsTbl: 存储商品信息(GoodsID, GoodsName等)。
- LinkTbl: 存储用户与商品的现有链接关系,字段包括 GoodsRef (引用GoodsTbl) 和 UserRef (引用UsersTbl)。
- NewLinkTbl: 用于创建新的用户与商品链接,同样包含 UserRef 和 GoodsRef 字段。
我们的目标是:在 NewLinkTbl 表单中,当 UserRef 字段被选中后,GoodsRef 字段的查找列表(Lookup List)应仅显示那些在 LinkTbl 中未与当前选定用户关联的商品。
2. 解决方案概述:高级引用限定符与脚本包含
ServiceNow提供了“引用限定符”(Reference Qualifier)功能来过滤引用字段的查找结果。对于需要根据表单中其他字段动态变化的复杂过滤逻辑,最有效的方法是使用“高级”(Advanced)引用限定符,并结合“脚本包含”(Script Include)来执行服务器端逻辑。
初始尝试使用客户端脚本(Client Script)通过 g_form.setValue() 来修改字段值,但这并不能直接过滤引用字段的查找列表。引用字段的查找列表过滤必须通过引用限定符来实现。
3. 实现步骤
3.1 配置引用字段的“高级引用限定符”
首先,我们需要修改 NewLinkTbl 表中 GoodsRef 字段的配置。
- 导航到字段定义: 在ServiceNow Studio中,找到 NewLinkTbl 表,然后定位到 GoodsRef 字段。
- 设置引用限定符类型:
- 点击 GoodsRef 字段,打开其配置面板。
- 将“使用引用限定符”(Use reference qualifier)设置为“高级”(Advanced)。
- 输入引用限定符脚本: 在“引用限定符”(Reference qual)字段中,输入以下JavaScript代码:
javascript:(new ScriptIncludeJS()).getNotLinkedGoods(current.user_name);
- 解释:
- javascript::表明这是一个JavaScript表达式。
- (new ScriptIncludeJS()):创建 ScriptIncludeJS 脚本包含的一个实例。
- .getNotLinkedGoods(current.user_name):调用该实例中的 getNotLinkedGoods 方法,并将当前表单记录的 user_name 字段值作为参数传递。这里的 current.user_name 指的是 NewLinkTbl 表单中 UserRef 字段的实际数据库字段名(通常是 sys_id 或 name,根据配置)。为了清晰起见,假设 UserRef 字段的实际数据库字段名为 user_name。如果 UserRef 是引用字段,current.user_name 会传递其 sys_id。
- 解释:
3.2 创建或修改脚本包含(Script Include)
接下来,我们需要创建一个名为 ScriptIncludeJS 的脚本包含,并在其中实现 getNotLinkedGoods 方法。
创建脚本包含:
- 在ServiceNow Studio中,导航到“服务器开发” -> “脚本包含”。
- 点击“新建”创建一个新的脚本包含。
- 名称 (Name): ScriptIncludeJS
- 客户端可调用 (Client callable): 勾选此项,因为引用限定符会从客户端调用它。
- 可从以下范围访问 (Accessible from): 设置为“所有应用程序范围”或您的应用程序范围。
编写脚本: 在脚本字段中输入以下代码:
var ScriptIncludeJS= Class.create(); ScriptIncludeJS.prototype = Object.extend(global.AbstractAjaxProcessor, { /** * 获取未与指定用户关联的商品列表的引用限定符字符串。 * * @param {string} userID - 当前选定用户的sys_id。 * @returns {string} - 一个GlideRecord查询字符串,用于过滤GoodsTbl。 */ getNotLinkedGoods: function(userID) { var linkedGoodsSysIDs = []; // 用于存储已关联商品的sys_id // 1. 验证传入的userID是否有效 if (!userID) { return 'sys_idISNOTEMPTY'; // 如果没有选择用户,则显示所有商品 } // 2. 查询LinkTbl,找到所有已与该用户关联的商品 // 假设 'x_1234_pro_0_linkedtbl' 是 LinkTbl 的表名 // 假设 LinkTbl 中的用户引用字段名为 'user_name' (存储sys_id) // 假设 LinkTbl 中的商品引用字段名为 'goods_name' (存储sys_id) var grLink = new GlideRecord('x_1234_pro_0_linkedtbl'); // 替换为您的LinkTbl表名 grLink.addQuery('user_name', userID); // 查询与传入userID关联的记录 grLink.query(); while(grLink.next()) { // 将已关联商品的sys_id添加到数组中 linkedGoodsSysIDs.push(grLink.getValue('goods_name')); // 替换为LinkTbl中商品引用字段的数据库名 } // 3. 构建引用限定符字符串 if (linkedGoodsSysIDs.length === 0) { // 如果没有已关联的商品,则显示所有商品(即不进行过滤) return 'sys_idISNOTEMPTY'; } else { // 返回一个 'sys_id NOT IN' 查询字符串,排除已关联的商品 return 'sys_idNOT IN' + linkedGoodsSysIDs.join(','); } }, type: 'ScriptIncludeJS' });
- 代码解释:
- getNotLinkedGoods(userID) 方法接收 NewLinkTbl 表单中 UserRef 字段的 sys_id。
- linkedGoodsSysIDs 数组用于收集所有已与该 userID 关联的商品的 sys_id。
- new GlideRecord('x_1234_pro_0_linkedtbl'):创建一个 GlideRecord 对象来查询 LinkTbl(请务必替换为您的实际 LinkTbl 表名)。
- grLink.addQuery('user_name', userID):根据传入的 userID 过滤 LinkTbl 中的记录。请确保 'user_name' 是 LinkTbl 中引用 UsersTbl 的字段的数据库名称。
- grLink.getValue('goods_name'):获取 LinkTbl 中引用 GoodsTbl 的字段的 sys_id。请确保 'goods_name' 是 LinkTbl 中商品引用字段的数据库名称。
- return 'sys_idNOT IN' + linkedGoodsSysIDs.join(','):这是关键部分。它返回一个标准的GlideRecord查询字符串。当 GoodsRef 字段的查找列表被打开时,ServiceNow会使用这个字符串来过滤 GoodsTbl,只显示那些 sys_id 不在 linkedGoodsSysIDs 列表中的商品。
- sys_idISNOTEMPTY:这是一个通用的查询,表示 sys_id 不为空,即显示所有记录。当没有已关联商品时,返回此值以确保所有商品都可选。
- 代码解释:
4. 效果验证
完成上述配置后:
- 在 NewLinkTbl 表单中,首先选择一个“用户名称”(UserRef)。
- 然后,点击“商品名称”(GoodsRef)字段的查找图标。
- 此时,弹出的商品列表将只会显示那些尚未与您在第一步中选择的用户关联的商品。
5. 注意事项与最佳实践
- 表名和字段名准确性: 在 ScriptIncludeJS 中使用的表名(如 x_1234_pro_0_linkedtbl)和字段名(如 user_name, goods_name)必须与您的ServiceNow实例中的实际数据库名称完全匹配。可以通过右键点击字段 -> “显示XML”或“配置字典”来确认。
- 性能考虑: 脚本包含在服务器端运行,通常比客户端脚本性能更优。但如果 LinkTbl 包含大量数据,并且 getNotLinkedGoods 方法执行的查询非常复杂,可能会对性能产生影响。确保查询条件优化,并考虑添加索引。
- 用户体验: 这种动态过滤机制显著提升了数据输入的准确性,减少了用户选择错误或重复关联的可能性。
- 错误处理: 在实际生产环境中,可以考虑在 ScriptIncludeJS 中添加更健壮的错误处理和日志记录机制,以便于调试和监控。
- 引用字段的显示值与实际值: current.user_name 如果 user_name 是引用字段,通常会传递其 sys_id。在 ScriptIncludeJS 中,如果您的 LinkTbl 中的 user_name 字段存储的是 sys_id,则直接使用即可。如果存储的是 UserName 字符串,则需要调整查询逻辑。本教程假设 user_name 和 goods_name 在 LinkTbl 中存储的是引用记录的 sys_id。
6. 总结
通过巧妙地结合ServiceNow的高级引用限定符和脚本包含,我们可以实现复杂且动态的引用字段过滤逻辑。这种方法不仅保证了数据的一致性和准确性,还极大地优化了用户在表单中进行数据选择时的体验,是ServiceNow平台开发中处理关联数据过滤的强大工具。
文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《动态过滤引用字段:关联数据与选择联动》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
165 收藏
-
310 收藏
-
472 收藏
-
329 收藏
-
102 收藏
-
239 收藏
-
254 收藏
-
233 收藏
-
303 收藏
-
141 收藏
-
102 收藏
-
313 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 514次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 499次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习