IDEA Web渲染插件开发(二)— 自定义JsDialog
《IDEA Web渲染插件开发(一)》中,我们了解到了如何编写一款用于显示网页的插件,所需要的核心知识点就是IDEA插件开发和JCEF,在本文中,我们将继续插件的开发,为该插件的JS Dialog显示进行自定义处理。
背景
在开发之前,我们首先要了解下什么是JS Dialog。有过Web页面开发经历的开发者都或多或少使用过这样一个JS的API:alert('this is a message')
,当JS页面执行这段脚本的时候,在浏览器上会有类似于如下的显示:
同样,当我们使用confirm('ok?')
的时候,会显示如下:
以及,使用prompt(input your name: ')
,有如下的显示:
这些弹框一般来说都是原生的窗体,例如,当我们在之前的《IDEA Web渲染插件开发(一)》中的Web渲染插件来打开上面的Demo网页的时候,效果如下:
alert
confirm
prompt
可以看到,原生窗体显得不是那么好看。那么,我们能不能自定义这个原生窗体呢?答案是肯定的,接下来就要用到JCEF里面一个Handler CefJSDialogHandler(java-cef/CefJSDialogHandler)。
CefJSDialogHandler
对于该Handler,官方注释为:
Implement this interface to handle events related to JavaScript dialogs. The methods of this class will be called on the UI thread.
实现此接口以处理与JavaScript对话框相关的事件。将在UI线程上调用此类的方法。
对于该Handler,里面有一个核心的接口方法:
/**
* Called to run a JavaScript dialog. Set suppress_message to true and
* return false to suppress the message (suppressing messages is preferable
* to immediately executing the callback as this is used to detect presumably
* malicious behavior like spamming alert messages in onbeforeunload). Set
* suppress_message to false and return false to use the default
* implementation (the default implementation will show one modal dialog at a
* time and suppress any additional dialog requests until the displayed dialog
* is dismissed). Return true if the application will use a custom dialog or
* if the callback has been executed immediately. Custom dialogs may be either
* modal or modeless. If a custom dialog is used the application must execute
* callback once the custom dialog is dismissed.
*
* @param browser The corresponding browser.
* @param origin_url The originating url.
* @param dialog_type the dialog type.
* @param message_text the text to be displayed.
* @param default_prompt_text value will be specified for prompt dialogs only.
* @param callback execute callback once the custom dialog is dismissed.
* @param suppress_message set to true to suppress displaying the message.
* @return false to use the default dialog implementation. Return true if the
* application will use a custom dialog.
*/
public boolean onJSDialog(CefBrowser browser, String origin_url, JSDialogType dialog_type,
String message_text, String default_prompt_text, CefJSDialogCallback callback,
BoolRef suppress_message);
注释翻译如下:
在调用一个JS的Dialog的时候会调用该方法。设置
suppress_message
为true
并使该方法返回false
来抑制这个消息(抑制消息比立即执行回调更可取,因为它用于检测可能的恶意行为,如onbeforeunload中的垃圾邮件警报消息)。设置suppress_message
为false
并且返回false
来使用默认的实现(默认的实现将会立刻展示一个模态对话框并抑制任何额外的对话框请求直到当前展示的对话框已经销毁)。如果应用程序想要使用一个自定义的对话框或是回调callback已经立刻被执行了,则返回true
。自定义的对话框可以是模态或是非模态的。如果使用了一个自定义的对话框,那么一旦自定义对话框销毁后,应用程序需要立即执行回调。
首先,我们编写类JsDialogHandler,实现该接口:
package com.compilemind.demo.handler;
import org.cef.browser.CefBrowser;
import org.cef.callback.CefJSDialogCallback;
import org.cef.handler.CefJSDialogHandler;
import org.cef.misc.BoolRef;
import static org.cef.handler.CefJSDialogHandler.JSDialogType.*;
public class JsDialogHandler implements CefJSDialogHandler {
@Override
public boolean onJSDialog(CefBrowser browser,
java.lang.String origin_url,
CefJSDialogHandler.JSDialogType dialog_type,
java.lang.String message_text,
java.lang.String default_prompt_text,
CefJSDialogCallback callback,
BoolRef suppress_message) {
// 具体内容见下文
}
@Override
public boolean onBeforeUnloadDialog(CefBrowser cefBrowser, String s, boolean b, CefJSDialogCallback cefJSDialogCallback) {
return false;
}
@Override
public void onResetDialogState(CefBrowser cefBrowser) {
}
@Override
public void onDialogClosed(CefBrowser cefBrowser) {
}
}
除了onJSDialog
方法,其他的我们暂时不关心,使用默认的处理。对于onJSDialog
的方法,我们编写如下的内容:
@Override
public boolean onJSDialog(CefBrowser browser,
java.lang.String origin_url,
CefJSDialogHandler.JSDialogType dialog_type,
java.lang.String message_text,
java.lang.String default_prompt_text,
CefJSDialogCallback callback,
BoolRef suppress_message) {
// 不抑制消息
suppress_message.set(false);
if (dialog_type == JSDIALOGTYPE_ALERT) {
// alert 对话框
} else if (dialog_type == JSDIALOGTYPE_CONFIRM) {
// confirm 对话框
} else if (dialog_type == JSDIALOGTYPE_PROMPT) {
// prompt 对话框
} else {
// 默认处理,不过理论不会进入这一步
return false;
}
// 返回true,表明自行处理
return false;
}
接下来,我们向CefBrowser进行注册(MyWebToolWindowContent类的构造函数中):
// 创建 JBCefBrowser
JBCefBrowser jbCefBrowser = new JBCefBrowser();
// 注册我们的Handler
jbCefBrowser.getJBCefClient()
.addJSDialogHandler(
new JsDialogHandler(),
jbCefBrowser.getCefBrowser());
// 将 JBCefBrowser 的UI控件设置到Panel中
this.content.add(jbCefBrowser.getComponent(), BorderLayout.CENTER);
至此,我们已经在该方法中对js的对话框类型进行了区分。接下来,就需要我们针对不同的对话框类型,展示不同的UI,那么需要我们了解如何在IDEA插件中弹出对话框。
IDEA插件对话框
DialogWrapper
DialogWrapper是IntelliJ下的所有对话框的基类,他并不是一个实际的UI控件,而是一个抽象类,在调用其show方法的时候,由IntelliJ框架进行展示。
Dialogs | IntelliJ Platform Plugin SDK (jetbrains.com)
我们需要做的就是编写一个类来继承该Wrapper。
AlertDialog
为了实现JS中的alert效果,我们首先编写AlertDialog:
import com.intellij.openapi.ui.DialogWrapper;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
public class AlertDialog extends DialogWrapper {
private final String content;
public AlertDialog(String title, String content) {
super(false);
setTitle(title);
this.content = content;
// init方法需要在所有的值设置到位的时候才进行调用
init();
}
@Override
protected @Nullable JComponent createCenterPanel() {
return new JLabel(this.content);
}
}
这个Dialog的实现非常的简单,通过构造函数传入对话框的title和content。其中,title在构造函数执行的时候,就通过DialogWrapper.setTitle(string)
完成设置;content赋值给AlertDialog的私有变量content,之后调用DialogWrapper.init()
方法进行初始化。
这里需要特别说明的是,init方法最好放在Dialog的私有变量赋值保存完成后才进行,因为init方法内部就会调用下面重写的createCenterPanel
方法。如果没有这样做,而是先init()
,再进行this.content = content
赋值,那么初始化的时候流程就是:
- 设置title。
- 调用init()。
- Init()内部调用
createCenterPanel()
。 - createCenterPanel返回一个空白的JLabel,因为此时
this.content
还是null。 - 进行
this.content = content
赋值操作。
最终弹出的对话框效果就是没有任何的内容,本人在这里也是踩了坑。
AlertDialog编写完成后,我们可以在需要的地方编写如下的代码进行弹框展示:
new AlertDialog("注意", "这是一个弹出框").show();
// 或
boolean isOk = new AlertDialog("注意", "这是一个弹出框").showAndGet();
于是,我们在之前的JSDialogHandler.onJSDialog中处理dialog_type == JSDIALOGTYPE_ALERT
的场景:
@Override
public boolean onJSDialog(CefBrowser browser,
java.lang.String origin_url,
CefJSDialogHandler.JSDialogType dialog_type,
java.lang.String message_text,
java.lang.String default_prompt_text,
CefJSDialogCallback callback,
BoolRef suppress_message) {
// 不抑制消息
suppress_message.set(false);
if (dialog_type == JSDIALOGTYPE_ALERT) {
// alert 对话框
new AlertDialog("注意", message_text).show();
return true;
}
return false;
}
问题处理
调试插件,当JS执行alert的时候,发现依然还是原生窗体。经过排查还会发现,问题情况如下:
- JS的alert依然是原生窗体。
- onJSDialog方法也进入了(可以使用断点或是控制台输出确认)。
- 控制台有异常:
Exception in thread "AWT-AppKit"
。
对于控制台的异常,详细如下:
Exception in thread "AWT-AppKit" com.intellij.openapi.diagnostic.RuntimeExceptionWithAttachments: EventQueue.isDispatchThread()=false Toolkit.getEventQueue()=com.intellij.ide.IdeEventQueue@fa771e7
对于EventQueue关键字的异常,有过GUI开发的读者应该很容易联想到应该是窗体事件消息机制的问题。
简单来说,窗体GUI的线程一般都是独立的,在这个线程中,会启动一个GUI事件队列循环,外部GUI输入(点击、拖动等等)会不断产生GUI事件对象,并按照一定的顺序进入事件循环队列,事件循环框架不断处理队列中的事件。对GUI的操作,比如修改窗体某个控件的文本或是想要对一个窗体进行模态显示,都需要在窗体GUI主线程进行,否则就会出现GUI的处理异常。
对于这类情况最常见问题场景就是:在窗体中点击一个按钮,点击后会单开一个线程异步加载大数据,加载完成后显示在窗体上。如果直接在加载大数据的线程中调用Form.setBigData()
(假如有这样一个设置文本的方法),一般来说就会出现异常:在非GUI线程中尝试修改GUI的相关值。在Java AWT中解决的方式,调用EventQueue.invokeLater(() -> { // do something} )
(异步)或是EventQueue.invokeAndWait(() -> { // do something} )
(同步)。调用之后,do something
就会被事件框架送入GUI线程执行了。
现在,我们回到一开始的问题,我们重新修改代码:
if (dialog_type == JSDIALOGTYPE_ALERT) {
// alert 对话框
EventQueue.invokeLater(() -> {
new AlertDialog("注意", message_text).show();
callback.Continue(true, "");
});
return true;
}
我们对代码进行断点确认线程,在onJSDialog执行的时候,所运行的线程是:AWT-AppKit
。
而EventQueue.invokeLater中所运行的线程是:AWT-EventQueue-0
,这个线程就是IDEA插件中的GUI线程。
修改线程处理后,让我们再次调用alert:
可以看到对话框已经显示为了使用IDEA插件下的dialog形式,但是这个dialog还不完全正确,一般的alert对话框,只会有一个确认按钮,而IDEA下的dialog默认是Cancel+OK的按钮组合。
Dialog按钮自定义(重写createActions)
IDEA插件的DialogWrapper默认情况下是Cancel+OK的按钮组合。那么如何自定义我们的按钮呢?可行的一种方式就是重写createActions。这个方法需要我们返回实现javax.swing.Action
接口的实例的数组,当然,IDEA插件也有对应的Wrapper:DialogWrapperAction。我们编写我们自己的OkAction:
protected class OkAction extends DialogWrapperAction {
public OkAction() {
super("确定");
}
@Override
protected void doAction(ActionEvent e) {
close(OK_EXIT_CODE);
}
}
务必注意,DialogWrapperAction的实现子类,必须是DialogWrapper的内部类,否则无法查看。
重新运行,查看AlertDialog的效果:
接下来,我们需要编写ConfirmDialog,来处理JS中的confirm。
ConfirmDialog
由于confirm天生需要取消和确定按钮,所以我们可以直接使用默认的DialogWrapper,不用重写Action的返回:
import com.intellij.openapi.ui.DialogWrapper;
import org.jetbrains.annotations.Nullable;
import javax.swing.*;
public class ConfirmDialog extends DialogWrapper {
private final String content;
public ConfirmDialog(String title, String content) {
super(false);
setTitle(title);
this.content = content;
// init方法需要在所有的值设置到位的时候才进行调用
init();
}
@Override
protected @Nullable JComponent createCenterPanel() {
return new JLabel(this.content);
}
}
在Handler中,我们对JSDIALOGTYPE_CONFIRM
分支进行:
if (dialog_type == JSDIALOGTYPE_CONFIRM) {
// confirm 对话框
EventQueue.invokeLater(() -> {
boolean isOk = new ConfirmDialog("注意", message_text).showAndGet();
callback.Continue(isOk, "");
});
return true;
}
这点和AlertDialog的差别在于,需要调用showAndGet
方法获取用户的点击是cancel还是ok的结果,使用callback返回给JS,才能使得JS的confirm调用获得正确的返回。下面是效果:
PromptDialog
对于PromptDialog,在对话框的界面,需要两个元素:文本提示和文本输入。同时,在对话框点击结束后,还需要获取用户的输入,代码如下:
public class PromptDialog extends DialogWrapper {
/**
* 显示信息
*/
private final String content;
/**
* 文本输入框
*/
private final JTextField jTextField;
public PromptDialog(String title, String content) {
super(false);
this.jTextField = new JTextField(10);
this.content = content;
setTitle(title);
// init方法需要在所有的值设置到位的时候才进行调用
init();
}
@Override
protected @Nullable JComponent createCenterPanel() {
// 2行1列的结构
JPanel jPanel = new JPanel(new GridLayout(2, 1));
jPanel.add(new JLabel(this.content));
jPanel.add(this.jTextField);
return jPanel;
}
public String getText() {
return this.jTextField.getText();
}
}
在这个类中,我们定义了一个私有字段JTextField
,之所以需要在类中持有该引用,是因为我们定义一个方法getText
,以便在对话框结束时,可以通过调用PromptDialog.getText
来获取用户输入。
编写完成后,我们在onJSDialog中对prompt类型的对话框进行处理:
if (dialog_type == JSDIALOGTYPE_PROMPT) {
// prompt 对话框
EventQueue.invokeLater(() -> {
PromptDialog promptDialog = new PromptDialog("注意", message_text);
boolean isOk = promptDialog.showAndGet();
String text = promptDialog.getText();
callback.Continue(isOk, text);
});
return true;
}
和之前不太一样的是,这里需要在showAndGet之后,调用getText来获取用户输入,并在callback.Continue(isOk, text)方法中
传入用户的数据数据。最终效果如下:
源码
w4ngzhen/intellij-jcef-plugin (github.com)
本次相关代码提交:support JsDialog
IDEA Web渲染插件开发(二)— 自定义JsDialog的更多相关文章
- IDEA Web渲染插件开发(一)— 使用JCEF
目前网上已经有了很多关于IDEA(IntelliJ平台)的插件开发教程了,本人觉得简书上这位作者秋水畏寒的关于插件开发的文章很不错,在我进行插件开发的过程中指导了我很多.但是综合下来看,在IDEA上加 ...
- HTML5 UI框架Kendo UI Web中如何创建自定义组件(二)
在前面的文章<HTML5 UI框架Kendo UI Web自定义组件(一)>中,对在Kendo UI Web中如何创建自定义组件作出了一些基础讲解,下面将继续前面的内容. 使用一个数据源 ...
- HTML5 UI框架Kendo UI Web教程:创建自定义组件(三)
Kendo UI Web包 含数百个创建HTML5 web app的必备元素,包括UI组件.数据源.验证.一个MVVM框架.主题.模板等.在前面的2篇文章<HTML5 Web app开发工具Ke ...
- Java Web高性能开发(二)
今日要闻: 性价比是个骗局: 对某个产品学上三五天个把月,然后就要花最少的钱买最多最好的东西占最大的便宜. 感谢万能的互联网,他顺利得手,顺便享受了智商上的无上满足以及居高临下的优越感--你们一千块买 ...
- Spring Boot 入门之 Web 篇(二)
原文地址:Spring Boot 入门之 Web 篇(二) 博客地址:http://www.extlight.com 一.前言 上一篇<Spring Boot 入门之基础篇(一)>介绍了 ...
- Redis总结(五)缓存雪崩和缓存穿透等问题 Web API系列(三)统一异常处理 C#总结(一)AutoResetEvent的使用介绍(用AutoResetEvent实现同步) C#总结(二)事件Event 介绍总结 C#总结(三)DataGridView增加全选列 Web API系列(二)接口安全和参数校验 RabbitMQ学习系列(六): RabbitMQ 高可用集群
Redis总结(五)缓存雪崩和缓存穿透等问题 前面讲过一些redis 缓存的使用和数据持久化.感兴趣的朋友可以看看之前的文章,http://www.cnblogs.com/zhangweizhon ...
- SpringBoot之WEB开发-专题二
SpringBoot之WEB开发-专题二 三.Web开发 3.1.静态资源访问 在我们开发Web应用的时候,需要引用大量的js.css.图片等静态资源. 默认配置 Spring Boot默认提供静态资 ...
- [转]通过继承ConfigurationSection,在web.config中增加自定义配置
本文转自:http://www.blue1000.com/bkhtml/2008-02/55810.htm 前几天写了一篇使用IConfigurationSectionHandler在web.conf ...
- SSMS2008插件开发(4)--自定义菜单
原文:SSMS2008插件开发(4)--自定义菜单 打开上次的项目MySSMSAddin中的Connect类,发现该类继于了两个接口:IDTExtensibility2和IDTCommandTarge ...
随机推荐
- java 日期格式化-- SimpleDateFormat 的使用。字符串转日期,日期转字符串
日期和时间格式由 日期和时间模式字符串 指定.在 日期和时间模式字符串 中,未加引号的字母 'A' 到 'Z' 和 'a' 到 'z' 被解释为模式字母,用来表示日期或时间字符串元素.文本可以使用单引 ...
- JAVA中的clone方法剖析
原文出自:http://blog.csdn.net/shootyou/article/details/3945221 java中也有这么一个概念,它可以让我们很方便的"制造"出一个 ...
- ☕【Java技术指南】「TestNG专题」单元测试框架之TestNG使用教程指南(上)
TestNG介绍 TestNG是Java中的一个测试框架, 类似于JUnit 和NUnit, 功能都差不多, 只是功能更加强大,使用也更方便. 详细使用说明请参考官方链接:https://testng ...
- Linkerd 2.10(Step by Step)—控制平面调试端点
Linkerd 2.10 系列 快速上手 Linkerd v2 Service Mesh(服务网格) 腾讯云 K8S 集群实战 Service Mesh-Linkerd2 & Traefik2 ...
- Win7/Win10+VS2017+OpenCV3.4.2安装、测试
安装VS2017 在微软官网https://www.microsoft.com,下载Visual Studio 2017安装包 用管理员权限运行vs2017 enterprise安装包,安装过程会持续 ...
- dotnet OpenXML 读取 PPT 内嵌 ole 格式 Excel 表格的信息
在 Office 中,可以在 PPT 里面插入表格,插入表格有好多不同的方法,对应 OpenXML 文档存储的更多不同的方式.本文来介绍如何读取 PPT 内嵌 ole 格式的 xls+ 表格的方法 在 ...
- tcmalloc jemalloc glibc内存分配管理模块性能测试对比
tcmalloc是谷歌提供的内存分配管理模块 jemalloc是FreeBSD提供的内存分配管理模块 glibc是Linux提供的内存分配管理模块 并发16个线程,分配压测3次,每次压15分钟,可以看 ...
- zigzag走线原理及应用
电路板上弯弯扭扭的走线有什么用 往期文章: 一文读懂高速互联的阻抗及反射(上) 一文读懂高速互联的阻抗及反射(中) 前面几篇文章有部分读者反馈太深奥,不好懂,要求来一点轻松易懂的.这不,它来了!本期文 ...
- 微信支付 V3 开发教程(一):初识 Senparc.Weixin.TenPayV3
前言 我在 9 年前发布了 Senparc.Weixin SDK 第一个开源版本,一直维护至今,如今 Stras 已经破 7K,这一路上得到了 .NET 社区的积极响应和支持,也受到了非常多的宝贵建议 ...
- Appium自动化(8) - 可定位的控件属性
如果你还想从头学起Appium,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1693896.html 前言 在前面几篇文章可以看到,一个 ...