金蝶云苍穹,新手初学者小白PC端java插件开发教学示例
<p><strong>注:</strong><br><strong>1.本文章的面向对象为刚刚接触金蝶云苍穹星空旗舰版,具有PC端Java插件开发需求的新手、小白、与初学者</strong><br>
<strong>2.本文章中涉及的开发环境包括:金蝶云苍穹星空旗舰版开发平台、安装了金蝶官方插件的IDEA</strong><br>
<strong>3.插件的实际开发涉及表单页面的生命周期,建议对此内容有一定熟悉,页面的生命周期参考文章:</strong><br>
https://vip.kingdee.com/article/403237280594265344?productLineId=29&lang=zh-CN<br>
<strong>4.本文档涉及一些金蝶官方的插件代码方法,这些插件代码方法的使用与介绍可以参考SDK文档:</strong><br>
https://dev.kingdee.com/sdk/Cosmic%20V5.0.002/index.html?nav=package<br>
<strong>5.文章的代码都是为了方便理解,使用的变量名都尽量简单,实际的业务开发中需要进行代码规范化处理,代码规范参考IDEA中金蝶官方插件的代码分析功能。</strong><br>
<strong>6.文章或许哪里讲的不够详细,或者存在疑意,有任何不明白或者不清楚的地方可以在评论区留言,如果我清楚的话会尽量尽快回复的</strong><br>
<strong>7.此文章在金蝶官方的开发者社区也有发布,本着能对像我一样熟悉博客园的金蝶开发者起到帮助的想法,所以将文章在园子又发布了一遍,金蝶开发者社区的此文链接:</strong><br>
https://vip.kingdee.com/article/799643532909519872?productLineId=29&lang=zh-CN</p>
<p><strong>一、插件开发思路</strong><br>
对于插件的开发,我们首先要确定的,是插件要实现的功能。<br>
然后根据功能,我们再进一步确定,这个功能包含的事件,事件触发页面的页面类型,事件的触发时机,事件的逻辑。<br>
等到这些都确定完后,我们就可以进行实际开发了。<br>
比如说,我想要开发一个表单插件,插件的功能是批量修改列表某些分录(数据行)的数量字段或者状态字段的值。<br>
那么这时候我会确定,这个功能的流程是:用户选中数据——>点击按钮——>修改选中数据的字段值——>字段列刷新。<br>
这样一来就可以确定,事件触发页面的页面类型(单据列表类型),触发时机(用户点击按钮之后),事件的逻辑(获取到用户选择的数据之后批量修改数据的字段值,最后刷新页面)。<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116090724230-1603172215.png"></p>
<p><strong>二、新建插件类</strong><br>
在IDEA中新建一个项目,在项目的对应plugin路径下,新建一个插件类,新建时选择继承的插件类名与继承的插件类型,例如:<br>
1.我的插件类为页面插件,所以选择plugin目录下的form文件夹新建插件类(右键点击form文件夹,点击新建,选择金蝶插件的继承插件)<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116090916813-1344644809.png"><br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116090927659-1404230642.png"></p>
<p>2.命名插件类并选择继承的插件类类型,这里我是列表页面插件,于是我选择标准单据列表插件类型<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116090938716-1692996360.png"></p>
<p>新建后的插件类内容:</p>
<pre><code>package ***.plugin.other;
import kd.bos.list.plugin.AbstractListPlugin;
import kd.sdk.plugin.Plugin;
/**
* 标准单据列表插件
*/
public class teacherListPlugin extends AbstractListPlugin implements PLugin {
}
</code></pre>
<p>3.编写代码实现事件的逻辑<br>
3.1,事件按钮添加<br>
因为我的插件事件需要用户点击按钮来触发,所以我在页面上添加了两个新的按钮(状态正常按钮与数量0按钮),<br>
并将按钮的操作代码选择为dothing(空操作),操作后刷新字段一个选择为数量(qty),一个选择为(status)<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091318187-142345391.png"><br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091327122-594205527.png"></p>
<p>3.2,编写代码<br>
因为我的事件在用户点击对应按钮后触发,所以首先注册两个按钮的点击监听事件<br>
(添加工具栏按钮点击监听的事件为addItemClickListeners,在生命周期registerListener中添加)</p>
<pre><code>private static final String FID_TOOLBAR = "toolbar";//默认的工具栏标识
@Override
public void registerListener(EventObject e) {
super.registerListener(e);
this.addItemClickListeners(FID_TOOLBAR);// 添加工具栏按钮点击的监听事件,监听的工具栏为字段标识对应的工具栏
}
</code></pre>
<p>插件类全代码:</p>
<pre><code>package ***.plugin.other;
import kd.bos.list.plugin.AbstractListPlugin;
import kd.sdk.plugin.Plugin;
import java.util.EventObject;
/**
* 标准单据列表插件
*/
public class teacherListPlugin extends AbstractListPlugin implements Plugin {
private static final String toolbar = "toolbar";//默认的工具栏标识
@Override
public void registerListener(EventObject e) {
super.registerListener(e);
this.addItemClickListeners(toolbar);// 添加工具栏按钮点击的监听事件,监听的工具栏为字段标识对应的工具栏
}
}
</code></pre>
<p>然后注册两个按钮的监听事件:</p>
<pre><code>private static final String statusbtn = "statusbtn";//状态正常按钮的对应字段标识
private static final String qtybtn = "qtybtn";//数量0按钮的对应字段标识
@Override
public void itemClick(ItemClickEvent evt) {
if (statusbtn.equals(evt.getItemKey())) {//监听状态正常按钮的点击事件
status();//状态正常事件逻辑处理方法
} else if (qtybtn.equals(evt.getItemKey())) {//监听数量0按钮的点击事件
qty();//数量重置为0事件逻辑处理方法
}
}
private void status(){//状态正常事件逻辑
}
private void qty(){//数量0事件逻辑
}
</code></pre>
<p>插件类全部代码:</p>
<pre><code>package ***.plugin.other;
import kd.bos.form.control.events.ItemClickEvent;
import kd.bos.list.plugin.AbstractListPlugin;
import kd.sdk.plugin.Plugin;
import java.util.EventObject;
/**
* 标准单据列表插件
*/
public class teacherListPlugin extends AbstractListPlugin implements Plugin {
private static final String toolbar = "toolbar";//默认的工具栏标识
private static final String statusbtn = "statusbtn";//状态正常按钮的对应字段标识
private static final String qtybtn = "qtybtn";//数量0按钮的对应字段标识
@Override
public void registerListener(EventObject e) {
super.registerListener(e);
this.addItemClickListeners(toolbar);// 添加工具栏按钮点击的监听事件,监听的工具栏为字段标识对应的工具栏
}
@Override
public void itemClick(ItemClickEvent evt) {
if (statusbtn.equals(evt.getItemKey())) {//监听状态正常按钮的点击事件
status();//状态正常事件逻辑处理方法
} else if (qtybtn.equals(evt.getItemKey())) {//监听数量0按钮的点击事件
qty();//数量重置为0事件逻辑处理方法
}
}
private void status(){//状态正常事件逻辑
}
private void qty(){//数量0事件逻辑
}
}
</code></pre>
<p>然后开始编写事件逻辑方法,(补充:状态字段的字段标识为status,控件类型为下拉列表,状态正常的对应值为"0",数量字段的字段标识为qty,控件类型为整数)<br>
状态正常事件:</p>
<pre><code>private void status(){//状态正常事件逻辑
ListSelectedRowCollection selectedRows = this.getSelectedRows();//获取当前列表选中行的索引
if (selectedRows == null || selectedRows.isEmpty()) {//校验选中行数量是否为空,为空则停止事件并提示
this.getView().showTipNotification("请先选择要操作的数据");
return;
}
//根据选中行的索引获取对应数据的主键值
List<Long> ids = new ArrayList<>();
for(ListSelectedRow row : selectedRows){
ids.add((Long)row.getPrimaryKeyValue());
}
//根据获取到的对应事件主键值批量查询获取数据
DynamicObject[] datas = BusinessDataServiceHelper.load("teacher",//第一个参数:要查询数据的表单的标识
"id,number,name,status",//第二个参数:要查询的数据的属性名
new QFilter[]{new QFilter("id", QCP.in,ids)}//第三个参数:要查询的数据的过滤条件
);
//遍历数据数组,对数据的对应字段的值进行修改
for (DynamicObject data : datas){
data.set("status","0");//正常状态的对应值为"0"
}
SaveServiceHelper.save(datas);//保存修改后的结果
this.getView().invokeOperation("refresh");//刷新页面
}
</code></pre>
<p>数量0事件:</p>
<pre><code>ListSelectedRowCollection selectedRows = this.getSelectedRows();//获取当前列表选中行的索引
if (selectedRows == null || selectedRows.isEmpty()) {//校验选中行数量是否为空,为空则停止事件并提示
this.getView().showTipNotification("请先选择要操作的数据");
return;
}
//根据选中行的索引获取对应数据的主键值
List<Long> ids = new ArrayList<>();
for(ListSelectedRow row : selectedRows){
ids.add((Long)row.getPrimaryKeyValue());
}
//根据获取到的对应事件主键值批量查询获取数据
DynamicObject[] datas = BusinessDataServiceHelper.load("teacher",//第一个参数:要查询数据的表单的标识
"id,number,name,qty",//第二个参数:要查询的数据的属性名
new QFilter[]{new QFilter("id", QCP.in,ids)}//第三个参数:要查询的数据的过滤条件
);
//遍历数据数组,对数据的对应字段的值进行修改
for (DynamicObject data : datas){
data.set("qty",0);//正常状态的对应值为"0"
}
SaveServiceHelper.save(datas);//保存修改后的结果
this.getView().invokeOperation("refresh");//刷新页面
</code></pre>
<p>插件类全代码:</p>
<pre><code>package ***.plugin.other;
import kd.bos.dataentity.entity.DynamicObject;
import kd.bos.entity.datamodel.ListSelectedRow;
import kd.bos.entity.datamodel.ListSelectedRowCollection;
import kd.bos.form.control.events.ItemClickEvent;
import kd.bos.list.plugin.AbstractListPlugin;
import kd.bos.orm.query.QCP;
import kd.bos.orm.query.QFilter;
import kd.bos.servicehelper.BusinessDataServiceHelper;
import kd.bos.servicehelper.operation.SaveServiceHelper;
import kd.sdk.plugin.Plugin;
import java.util.ArrayList;
import java.util.EventObject;
import java.util.List;
/**
* 标准单据列表插件
*/
public class teacherListPlugin extends AbstractListPlugin implements Plugin {
private static final String toolbar = "toolbar";//默认的工具栏标识
private static final String statusbtn = "statusbtn";//状态正常按钮的对应字段标识
private static final String qtybtn = "qtybtn";//数量0按钮的对应字段标识
@Override
public void registerListener(EventObject e) {
super.registerListener(e);
this.addItemClickListeners(toolbar);// 添加工具栏按钮点击的监听事件,监听的工具栏为字段标识对应的工具栏
}
@Override
public void itemClick(ItemClickEvent evt) {
if (statusbtn.equals(evt.getItemKey())) {//监听状态正常按钮的点击事件
status();//状态正常事件逻辑处理方法
} else if (qtybtn.equals(evt.getItemKey())) {//监听数量0按钮的点击事件
qty();//数量重置为0事件逻辑处理方法
}
}
private void status(){//状态正常事件逻辑
ListSelectedRowCollection selectedRows = this.getSelectedRows();//获取当前列表选中行的索引
if (selectedRows == null || selectedRows.isEmpty()) {//校验选中行数量是否为空,为空则停止事件并提示
this.getView().showTipNotification("请先选择要操作的数据");
return;
}
//根据选中行的索引获取对应数据的主键值
List<Long> ids = new ArrayList<>();
for(ListSelectedRow row : selectedRows){
ids.add((Long)row.getPrimaryKeyValue());
}
//根据获取到的对应事件主键值批量查询获取数据
DynamicObject[] datas = BusinessDataServiceHelper.load("teacher",//第一个参数:要查询数据的表单的标识
"id,number,name,status",//第二个参数:要查询的数据的属性名
new QFilter[]{new QFilter("id", QCP.in,ids)}//第三个参数:要查询的数据的过滤条件
);
//遍历数据数组,对数据的对应字段的值进行修改
for (DynamicObject data : datas){
data.set("status","0");//正常状态的对应值为"0"
}
SaveServiceHelper.save(datas);//保存修改后的结果
this.getView().invokeOperation("refresh");//刷新页面
}
private void qty(){//数量0事件逻辑
ListSelectedRowCollection selectedRows = this.getSelectedRows();//获取当前列表选中行的索引
if (selectedRows == null || selectedRows.isEmpty()) {//校验选中行数量是否为空,为空则停止事件并提示
this.getView().showTipNotification("请先选择要操作的数据");
return;
}
//根据选中行的索引获取对应数据的主键值
List<Long> ids = new ArrayList<>();
for(ListSelectedRow row : selectedRows){
ids.add((Long)row.getPrimaryKeyValue());
}
//根据获取到的对应事件主键值批量查询获取数据
DynamicObject[] datas = BusinessDataServiceHelper.load("teacher",//第一个参数:要查询数据的表单的标识
"id,number,name,qty",//第二个参数:要查询的数据的属性名
new QFilter[]{new QFilter("id", QCP.in,ids)}//第三个参数:要查询的数据的过滤条件
);
//遍历数据数组,对数据的对应字段的值进行修改
for (DynamicObject data : datas){
data.set("qty",0);//正常状态的对应值为"0"
}
SaveServiceHelper.save(datas);//保存修改后的结果
this.getView().invokeOperation("refresh");//刷新页面
}
}
</code></pre>
<p>3.3,注册插件<br>
插件类编写完成后,需要将代码绑定到对应页面上才能生效:<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091630439-1114277006.png"><br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091637409-1598268619.png"></p>
<p>示例:<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091646116-192463489.png"></p>
<p>然后点击保存页面<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091655847-1617777485.png"></p>
<p>三、页面运行测试插件<br>
点击页面预览按钮进入预览模式,然后测试对应功能<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091717891-1238174192.png"></p>
<p>我的:<br>
点击状态正常按钮前与点击正常状态按钮后:<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091735086-613530064.png"><br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091746394-1356303451.png"></p>
<p>点击数量0前与点击数量0按钮后:<br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091800546-275681006.png"><br>
<img src="https://img2024.cnblogs.com/blog/3567775/202601/3567775-20260116091812782-43383202.png"></p><br><br>
来源:https://www.cnblogs.com/chenxvhua/p/19491762
頁:
[1]