SmartBI拓展包二开入门开发

news2024/9/22 8:30:49

前言

新接到一个项目拓展包三开的需求,没有相关经验,学习开发,本文尝试通过简单的定位以及指导,确定修改点
SmartBI帮助文档-拓展包开发

登录

http://localhost:18080/smartbi/vision/index.jsp

在这里插入图片描述

后台配置

上传拓展包,先看看以前有什么功能

http://localhost:18080/smartbi/vision/config.jsp

在这里插入图片描述
指定位置可以看到对应的拓展包
在这里插入图片描述

重启

上传拓展包后 服务需要重启,也可以在WIN视窗找到对应的服务图标
在这里插入图片描述
在这里插入图片描述

重新登录

登录后,因为我做的是钉钉二开,我知道东西大概在哪里放着,所以寻找,自己根据自己需求来找对应页面,我目前功能需要修改这两个名字

在这里插入图片描述

F12定位

定位当前页面使用JS在那个位置,因为SmartBI依靠的是复写追加模式
在这里插入图片描述

页面元素定位

通过页面元素检索
在这里插入图片描述

前端代码

在这里插入图片描述

搜索关键字

我们要修改这个按钮名字:PushPreview,发现他是一个语言包用UNICODE转码的,那我们把要修改的字段配置值修改下即可,在拓展包-直接打开-找到对应位置-修改好-重新上传拓展包-覆盖理论即可解决,找个在线转码翻译下
在这里插入图片描述

重复上传

在这里插入图片描述

原因

因为SmartBI加载拓展包后会解压拓展包形成一个目录,这个目录保留了本次变更,同名拓展包上传并没有覆盖这个地方;
每个拓展包实际上就是一个war包,但这不是主要原因,上传了拓展包会覆盖原有的,这没问题,问题在在于浏览器,虽然我们清除了缓存,但是因为SmartBI使用的是JSP页面,导致删不干净

E:\smartbi\Tomcat\bin\exts-smartbi

在这里插入图片描述

解决

使用无痕浏览器,重新加载
完成一个需求
在这里插入图片描述

代码

放一段修改代码,看下钉钉安装包开发逻辑,入门

 var DingDingPushPanel = jsloader.resolve("bof.schedule.panel.DingDingPushPanel");

TaskCustom.prototype.initTaskType_DingDingPushTask = TaskCustom.prototype.initTaskType; 

//初始化任务类型
TaskCustom.prototype.initTaskType = function(isNew){
	this.initTaskType_DingDingPushTask.apply(this, arguments);
	this.cb_taskType.insertItem("DINGDING_PUSH", "${DingDingPush}");
//	this.cb_taskType.setSelectedItem("DINGDING_PUSH", "${DingDingPush}");  // 先默认选择这个 方便开发测试少点一次
}


//初始化不同类型下 界面
TaskCustom.prototype._DingDingPushTask_initPanel = TaskCustom.prototype.initPanel;
TaskCustom.prototype.initPanel = function(){
	if (this.cb_taskType.getSelectedId() == "DINGDING_PUSH") {
		this.initDingDingPushPanel();
	} else {
		this._DingDingPushTask_initPanel.apply(this, arguments);
	}
}

TaskCustom.prototype._DingDingPushTask_elemTaskType_change_handler = TaskCustom.prototype.elemTaskType_change_handler;
//根据任务类型进行不同类型任务界面切换
TaskCustom.prototype.elemTaskType_change_handler = function(obj,oldSelectedId, newSelectedId, oldTitle, newTitle) {
	if (newSelectedId == "DINGDING_PUSH") {  
		this.setTaskPanel("DINGDING_PUSH");
	} else {
		this._DingDingPushTask_elemTaskType_change_handler.apply(this, arguments);
	}
}
//界面变化
TaskCustom.prototype._DingDingPushTask_setTaskPanel = TaskCustom.prototype.setTaskPanel;
TaskCustom.prototype.setTaskPanel = function(type){
	this._DingDingPushTask_setTaskPanel.apply(this, arguments);
	if (this.dingDingPushPanel) {
		this.dingDingPushPanel.hide();
	} 
	if (type == "DINGDING_PUSH") {
        //三开 临时修改变量名称
        this.elemButtonTest.value ="推送正式群";
		if (this.dingDingPushPanel) {
			this.dingDingPushPanel.show();
		} else {
			this.initDingDingPushPanel();
		}
	} else{
	//ExtJumpReport
	this.elemButtonTest.value ="测试运行(T)";
	}

}

//初始化DINGDING_PUSH类型的界面
TaskCustom.prototype.initDingDingPushPanel = function(){
	this.dingDingPushPanel = new DingDingPushPanel(this.elemTasksDiv,this.oldTaskInfo);
	this.dingDingPushPanel.onPushPreview.subscribe(this.ddPushPreview, this);
}

//任务信息
TaskCustom.prototype._DingDingPushTask_getTaskInfo = TaskCustom.prototype.getTaskInfo;
TaskCustom.prototype.getTaskInfo = function(){
	var taskInfo = this._DingDingPushTask_getTaskInfo.apply(this, arguments);
	if (!taskInfo) {
		return taskInfo;
	}
	var msg;
	if(this.cb_taskType.getSelectedId() == "DINGDING_PUSH") { 
		var ddPushInfo = this.dingDingPushPanel.getValue();
		var msg = this.dingDingPushPanel.validate(ddPushInfo);
		if (msg) {
			alert(msg);
			return false;
		} 		
		this.taskInfo.ddPushInfo = ddPushInfo;
	}
	return this.taskInfo;
}

// 保存
TaskCustom.prototype._DingDingPushTask_doSave = TaskCustom.prototype.doSave ;
TaskCustom.prototype.doSave = function(parentId) {
	parentId = parentId || this.parentFolderId;
	if (this.taskInfo.type == "DINGDING_PUSH") {
		if (this.dingDingPushTaskSave(parentId)){
			this.DefaultOptionType = this.OptionType_MODIFY;
			this.elemButtonTest.disabled = false;
			this.copySaveToCache();
			this.elemTaskName.disabled = true;
			this.cb_taskType.setEnabled(false);
			this.doOnClose();
		}
	} else {
		this._DingDingPushTask_doSave(parentId);
	}
}

/**
 * 保存操作
 */
TaskCustom.prototype.dingDingPushTaskSave = function(parentId){
	if (this.DefaultOptionType == this.OptionType_ADD) {
		var ret = util.remoteInvokeEx("CatalogService", "isCatalogElementAccessible", [ parentId, "WRITE" ]);
		if (ret && ret.succeeded) {
			if (!ret.result) {
				alert("${Youdonothavetheresources}${Leftdoublequote}${Toedit}${Rightdoublequote}${Permission}${Exclamation}\n" + parentId);
				return false;
			}
		} else {
			return false;
		}
	}
	var ret = false;
	if(this.taskInfo.taskAlias == ""){
		this.taskInfo.taskAlias = this.taskInfo.taskName;
		this.elemTaskAlias.value = this.taskInfo.taskName;
	}
	// clone一个新对象 并把参数设置分割出来 后面分开处理
	if (this.taskInfo.type == "DINGDING_PUSH") {// 报表订阅
		if (this.DefaultOptionType == this.OptionType_ADD) {// 任务新增
			ret = util.remoteInvokeEx("DDPushTaskModule",
					"createDDPushTask", [ this.taskInfo.taskName,
							this.taskInfo.taskAlias, this.taskInfo.taskNote,
							this.taskInfo.ddPushInfo, parentId ]);
		} else if (this.DefaultOptionType == this.OptionType_MODIFY) {// 任务修改
			ret = util.remoteInvokeEx("DDPushTaskModule",
					"updateDDPushTask", [ this.taskId,
							this.taskInfo.taskName, this.taskInfo.taskAlias,
							this.taskInfo.taskNote, this.taskInfo.ddPushInfo]);
		}
	} 
		
	if (ret && ret.succeeded && ret.result) {	
		this.taskId = ret.result;
		alert("${Theoperationissuccessful}${Exclamation}");
		this.needRefresh.fire(this, parentId || ret.result);
		this.onUpdateInfo.fire(this, ret.result);
		return true;
	} 
}



/**
 * 修改任务的时候初始化面板
 */
//初始化其他的定制开发的任务类型
TaskCustom.prototype._DingDingPushTask_initOtherTask = TaskCustom.prototype.initOtherTask;
TaskCustom.prototype.initOtherTask = function(result) {	
	this._DingDingPushTask_initOtherTask.apply(this, arguments);
	this.setTaskPanel(result.type);
}


/**
 * 缓存修改时进入的定制信息。
 */
TaskCustom.prototype._DingDingPushTask_cacheModifyCustomInfo = TaskCustom.prototype.cacheModifyCustomInfo;
TaskCustom.prototype.cacheModifyCustomInfo = function(){
	this._DingDingPushTask_cacheModifyCustomInfo();
	if (this.oldTaskInfo.type == 'DINGDING_PUSH') {
		this.oldCustomInfo.ddPushInfo = $.extend(true, {}, this.oldTaskInfo.ddPushInfo);
	}
}

/**
* 推送预览 推送到测试群
*/
TaskCustom.prototype.ddPushPreview = function() {
	if(!this.getTaskInfo()) {
		return;
	}
	if (!this.taskInfo.ddPushInfo.testWebhook) {
		alert("【${DingDingSettings}】【${TestGroup}】${PleaseInputWebhookAddress}");
		return;
	}
	if (!this.taskInfo.ddPushInfo.testSignSecret) {
		alert("【${DingDingSettings}】【${TestGroup}】${PleaseInputSignSecret}!");
		return;
	}
	var taskInfo = $.extend(true, {}, this.taskInfo);
	taskInfo.ddPushInfo.webhook = taskInfo.ddPushInfo.testWebhook;
	taskInfo.ddPushInfo.signSecret = taskInfo.ddPushInfo.testSignSecret;
	taskInfo.currentUserName = registry.get("currentUserName")
	taskInfo.isPushPreview = true;
	taskInfo.ddPushInfo.isPushPreview = true;
	taskInfo.type = this.cb_taskType.getSelectedId();
	var dialogConfig = {};
	dialogConfig.title = "${PushPreview}";
	dialogConfig.size = DialogFactory.getInstance().size.SMALL;
	dialogConfig.fullName = "bof.schedule.dialog.TestRunningDialog";
	DialogFactory.getInstance().showDialog(dialogConfig, taskInfo, null, this);
}

//全局销毁函数
TaskCustom.prototype._DingDingPushTask_destroy  = TaskCustom.prototype.destroy;
TaskCustom.prototype.destroy = function(){
	if(this.dingDingPushPanel){
		this.dingDingPushPanel.onPushPreview.unsubscribe(this.ddPushPreview, this);
		this.dingDingPushPanel.destroy();
	}
	this._DingDingPushTask_destroy();
}

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

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

相关文章

MySQL和Redis的数据一致性

MySQL和Redis的数据一致性 多线程环境下的涉及读写的缓存才会存在MySQL和Redis的数据不一致问题 先删除缓存再更新数据库再延时删除缓存 线程一删除缓存线程一更新数据线程二开始查数据如果第二步线程一更新数据延时,那么线程二会重新从数据库加载数据&#xff0…

超好用的windows系统工具PowerToys

文章目录 Github地址基本介绍使用 Github地址 PowerToys 基本介绍 是windows官方好用的工具箱,包括各种工具 使用 要带上win键 此工具安装后每次运行电脑自启动,桌面没有快捷方式,只能右下角 窗口在上效果演示,会被蓝线框到…

基于GeoTools使用JavaFx进行矢量数据可视化实战

目录 前言 一、JavaFx展示原理说明 二、GeoTools的Maven依赖问题 三、引入Geotools相关的资源包 四、创建JavaFx的Canvas实例 五、JavaFx的Scene和Node的绑定 六、总结 前言 众所周知,JavaFx是Java继Swing之后的又一款用于桌面应用的开发利器。当然&#xff0…

江科大/江协科技 STM32学习笔记P22

文章目录 AD单通道&AD多通道ADC基本结构和ADC有关的库函数AD单通道AD.cmain.c连续转换,非扫描模式的AD.c AD多通道AD.cmain.c AD单通道&AD多通道 ADC基本结构 第一步,开启RCC时钟,包括ADC和GPIO的时钟,ADCCLK的分频器也需…

openvidu私有化部署

openvidu私有化部署 简介 OpenVidu 是一个允许您实施实时应用程序的平台。您可以从头开始构建全新的 OpenVidu 应用程序,但将 OpenVidu 集成到您现有的应用程序中也非常容易。 OpenVidu 基于 WebRTC 技术,允许开发您可以想象的任何类型的用例&#xf…

回归预测|基于黏菌优化LightGBM的数据回归预测Matlab程序SMA-LightGBM 多特征输入单输出

回归预测|基于黏菌优化LightGBM的数据回归预测Matlab程序SMA-LightGBM 多特征输入单输出 文章目录 前言回归预测|基于黏菌优化LightGBM的数据回归预测Matlab程序 多特征输入单输出 SMA-LightGBM 一、SMA-LightGBM模型1. **LightGBM**2. **黏菌智能优化算法(SMA&…

知识中台是什么?它如何实现高效知识管理?

引言 在信息化浪潮席卷全球的今天,企业面临的不仅是市场的激烈竞争,更是知识爆炸带来的管理挑战。如何在浩瀚的信息海洋中提炼出有价值的知识,并将其快速转化为企业的核心竞争力,成为了每个企业必须深思的问题。在此背景下&#…

二叉树的重要概念

前言: 二叉树是树形结构的一个重要类型,一般的树也可以转化成二叉树来解决问题。在数据结构的系统中,树形结构也是信息存储和遍历的重要实现,二叉树的最大特点就是一个根包含着左右子树的形式,许多具有层次关系的问题…

单元测试注解:@ContextConfiguration

ContextConfiguration注解 ContextConfiguration注解主要用于在‌Spring框架中加载和配置Spring上下文,特别是在测试场景中。 它允许开发者指定要加载的配置文件或配置类的位置,以便在运行时或测试时能够正确地构建和初始化Spring上下文。 基本用途和工…

【开源社区】Elasticsearch(ES)中空值字段 null_value 及通过exists查找非空文档

文章目录 0、声明1、问题描述2、问题剖析2.1 NULL或者空值类型有哪些2.2 案例讲解:尝试检索值为 null 的字段2.3 解决思路 3、使用 null_value 的诸多坑(避免生产事故)3.1 null_value 替换的是索引,并不会直接替换源数据3.2 不支持…

LVS(Linux Virtual Server)详解

LVS(Linux Virtual Server)是一个用于负载均衡的开源软件项目,旨在通过集群技术实现高性能、高可用的服务器系统。它运行在Linux操作系统上,并且可以利用内核级的资源来提高性能和稳定性。 思维导图 LVS的工作原理 LVS主要基于Ne…

IDEA 2022.1.4用前需知

目录 一、配置国内源 二、正确再次创建新项目方式 IDEA 2022.1.4下载地址 一、配置国内源 1、查看本地仓库地址 2、设置国内源-添加Setting.xml文件内容 3、修改目录(考虑到当前硬盘空间大小,英文目录名) 1)创建你要移动过去…

xCat部署及分发操作系统

一、环境准备 此次安装部署均在VMware虚拟机上运行。系统采用通用稳定的centos7系统,移植到其他(linux)系统应该问题不大。软件服务器的VMware虚拟机的创建部分就跳过了. 1.1服务器的配置 IP主机名配置备注192.168.11.10master4C/8G/60GXcat/DNS/DHCP/NTP/TFTP192.168.11.11n…

【超音速专利 CN109636858A】锂电池涂布图像采集标定方法、系统、设备及存储介质

申请号CN201811276578.4公开号(公开)CN109636858A申请日2018.10.30申请人(公开)广州超音速自动化科技股份有限公司(超音速人工智能科技股份有限公司)发明人(公开)赵兵锁(张); 张俊峰(张); 梁土伟 相关术语…

读零信任网络:在不可信网络中构建安全系统14流量信任

1. 流量信任 1.1. 网络流的验证和授权是零信任网络至关重要的机制 1.2. 零信任并非完全偏离已知的安全机制,传统的网络过滤机制在零信任网络中仍然扮演着重要的角色 2. 加密和认证 2.1. 加密和认证通常是紧密相关的,尽管其目的截然不同 2.1.1. 加密提…

Spring Boot - 开启log-request-details详细记录调测Controller接口

文章目录 概述实现详细日志输出1. 调整日志级别2. 示例接口3. 启用请求详细信息日志 注意事项 概述 在Spring Boot项目中,调试Controller接口的请求和响应信息可以极大地帮助开发人员排查问题并确保应用程序的安全性和性能。 实现详细日志输出 1. 调整日志级别 …

在LabVIEW中高效读取大型CSV文件的方法

当尝试使用“读取分隔的电子表格VI”从大型CSV文件(数百MB)中读取数据时,可能会遇到内存已满错误。这是因为该VI会一次性读取整个文件并将其转换为数值数组,导致占用大量内存。 解决方案 可以使用“从文本文件VI读取”来部分读取…

Acrel-1000DP分布式光伏监控系统在光伏并网系统的实际应用分析-安科瑞 蒋静

摘要:为实现“双碳目标”即中国明确提出的2030年“碳达峰”与2060年“碳中和”目标。推动节能减排、实现经济可持续发展,我国采取了一系列方案和行动。其中就包括能源绿色低碳转型行动:大力发展新能源,‌推动煤电节能降碳&#xf…

golang 条件语句中有多条件时多条件的执行顺序和执行效率问题总结 -- if else条件语句 多条件时执行顺序为从左到右

在golang的 条件语句中有多个执行条件时,go的执行顺序为从左到右依次执行, 如果多个条件是 && 条件与的关系,则执行条件的顺序就与程序的效率密切相关, if else 的多个条件示例: func TestOrderTest(t *testi…

监听器——监听着我们WEB项目中的域对象

监听器 ——它监听着我们WEB项目中的域对象 何时被创建被销毁 ServlertContextListener——它监听ServletContext对象的创建和销毁 contextInitialized 创建的时候会调用 Tomcat启动时调用 contextDestroyed销毁的时候自动会调用的方法 Tomcat终止 这两个方法中携带的参数S…