AX有个不错的功能,在窗体上点击右键->记录信息 出来的窗体上就会有重命名这个选项,这个功能最终调用的是目的表的renamePrimaryKey方法,这个方法会更新该表的主键值,并更新引用了该主键的从表的值,比如调用CustTable的renamePrimaryKey方法,不仅会更新CustTable的AccountNum的值,也会更新CustTrans中AccoutNum的值。
有些窗体并没有的 记录信息 窗体里并没有显示这个选项,为什么那?记录信息这个快捷菜单调用的窗体是SysRecordInfo
在该窗体的init方法里就规定了哪些窗体不能显示 重命名这个功能。
关于重命名这个功能,最关键的代码如下:
common = formObjectSet.cursor();
dictTable = new SysDictTable(common.TableId);
dictField = new SysDictField(dictTable.id(), dictTable.primaryKeyField());
if (!dictField ||
common.TableId == tablenum(DataArea) ||
dictField.rights() < AccessType::Edit ||
(common.TableId == tablenum(UserInfo) && common.(fieldnum(UserInfo, Id)) == #AdminUser) ||
(common.TableId == tablenum(UserGroupInfo) && common.(fieldnum(UserGroupInfo, Id)) == #AdminUserGroup) ||
(common.TableId == tablenum(DomainInfo) && common.(fieldnum(DomainInfo, Id)) == #AdminDomain) ||
(common.TableId == tablenum(UserInfo) && !domainAccess()) ||
(common.TableId == tablenum(UserGroupInfo) && !domainAccess()) ||
(common.TableId == tablenum(DomainInfo) && !domainAccess())
// this line will disabled renaming of dimension
// || !dictTable.getIndexIdsOfUniqueOneFieldIndexes().elements()
)
{
renameGrp.visible(false);
merge.visible(false);
primaryKey.backgroundColor(4); // Menu background
}
else
{
primaryKey.label(dictField.label());
primaryKey.text(strfmt('%1', common.(dictTable.primaryKeyField())));
if (!common.(dictTable.primaryKeyField()))
renameGrp.visible(false);
}
dictTable = new SysDictTable(common.TableId);
dictField = new SysDictField(dictTable.id(), dictTable.primaryKeyField());
if (!dictField ||
common.TableId == tablenum(DataArea) ||
dictField.rights() < AccessType::Edit ||
(common.TableId == tablenum(UserInfo) && common.(fieldnum(UserInfo, Id)) == #AdminUser) ||
(common.TableId == tablenum(UserGroupInfo) && common.(fieldnum(UserGroupInfo, Id)) == #AdminUserGroup) ||
(common.TableId == tablenum(DomainInfo) && common.(fieldnum(DomainInfo, Id)) == #AdminDomain) ||
(common.TableId == tablenum(UserInfo) && !domainAccess()) ||
(common.TableId == tablenum(UserGroupInfo) && !domainAccess()) ||
(common.TableId == tablenum(DomainInfo) && !domainAccess())
// this line will disabled renaming of dimension
// || !dictTable.getIndexIdsOfUniqueOneFieldIndexes().elements()
)
{
renameGrp.visible(false);
merge.visible(false);
primaryKey.backgroundColor(4); // Menu background
}
else
{
primaryKey.label(dictField.label());
primaryKey.text(strfmt('%1', common.(dictTable.primaryKeyField())));
if (!common.(dictTable.primaryKeyField()))
renameGrp.visible(false);
}
我们可以看出对于一般窗体来说,最重要的是dictTable.primaryKeyField()这个方法的返回值,从这个方法的意思上看就是这个表要有主键,才能使用该功能。
如果认真测试一下这个方法的返回值是比较怪异的,按照我的理解,主键就是标识记录在数据库中记录的唯一性的字段。按照这样的理解,这个(些)字段应该是表属性PrimaryIndex对应的索引所包含的字段,因为在数据库表中的主键就是由它们组成的(当然还包含了DataAreaId,这里不讨论),这样primaryKeyField方法返回的无疑应该是这些字段,但实际上并不是这样的
1.primaryKeyField这个方法的返回值是某个字段的ID,也就是说对于组合主键,它只能返回其中的一个;
2.primaryKeyField这个方法的返回值并不是真实数据库表中对应主键字段(或者主键字段中的一个),只有在如下情况都满足时该方法才会有非零返回值
A).在表的Indexs节点含有一个包含这些字段的索引,并且其duplicate属性为false;
B).在表的Relations节点包含一个关联表为自身,且关联字段包含在A中索引字段的关系。
从上面的讨论不难看出,AX在数据和业务层面对主键有各自的定义,在数据库层面就是PrimaryIndex属性指定的索引所包含的字段,而在业务层面事通过关系来定义主键的,无论是在EDT上定义还是通过表的Relations接点(如果是组合主键,必须通过Realtion创建)定义。
当然在现有的系统中对于Main,Group,WorkSheetTable类型的表来说,这两个层面的主键定义基本都是一致的,但对于WorkSheetLine和Transactions之类的表就是另一回事了。
拿SalesLine这个表来说,其物理表主键为InventTransId(当然还要加上DataArreaId),但primaryKeyField函数的返回值为零,也就是说没有找到返回值,这时只要在Relations上添加关联表为自身SalesLine,关联字段为InventTransId的关系,该函数就会返回字段InventTransId对应的Id了。
当然由于没看到相关的文档描述,上面的描述这是我自己测试的结果,有不对的地方还望多多指正。
对于启用了本地化功能,并且采用了树状科目表结构的用户来说,会计科目表窗体的重命名功能也是没办法用的,因为:
if (common.tableId == tableNum(LedgerTable) && LedgerParameters::find().TreeStructure_CN)
{
renameGrp.visible(false);
TemplateGrp.visible(false);
}
这段代码将其屏蔽了。{
renameGrp.visible(false);
TemplateGrp.visible(false);
}
不太清楚为什么要把会计科目表的这个功能屏蔽掉。