chorme唤起Java开发的本地程序全踩坑记

背景说明

 在开发企业web应用时,往往需要进行订单通知,状态通知,或者需要一些插件式本地应用来扩展一些网页实现不了的功能等。以通知为例:如果网页标签页或者浏览器切出去了,意味着网页内部的通知是业务员无法感知到的。所以如何做到有效的系统通知功能。是需要去考虑的。2B的系统有一点比较好的是,往往可以限定系统以及浏览器,这也给开发减少了不少难度。那么读完本文您将学到什么。1.网页外的系统通知方案2.Java开发本地应用程序并形成可安装的exe文件。3.自定义本地浏览器协议

方案

一、通过浏览器自带的系统通知API--Notification对象来实现。
二、自己开发本地应用通过浏览器唤起

一、通过浏览器自带的系统通知API来实现

打开google浏览器,f12开启调试,在console控制台输入以下代码,并查看右下角。
function createNotify(title,options) {var PERMISSON_GRANTED = 'granted';var PERMISSON_DENIED = 'denied';var PERMISSON_DEFAULT = 'default';if (Notification.permission === PERMISSON_GRANTED) {notify(title,options);} else {Notification.requestPermission(function (res) {if (res === PERMISSON_GRANTED) {notify(title,options);}});}function notify($title,$options) {var notification = new Notification($title, $options);notification.onshow = function(event){ console.log('show : ',event); }notification.onclose = function(event){ console.log('close : ',event); }notification.onclick = function(event){ notification.close();}}
}createNotify('这是条愚人节通知',{body:'中国赢得了世界杯冠军'});
这种方案的特点在于:如果仅仅想做简单系统通知,那无疑该方案是最快最省成本的。缺点也比较明显:第一点,适配大部分浏览器及版本,但是依旧还有部分浏览器或者版本不支持。第二点,无法做比较酷炫的效果和自定义的其它功能。

二、自己开发本地应用

(1)方案选择

首先在界面应用开发的技术选型:1.C++ MFC or Qt2.Go Walk3.Java JSwing4.Javascript  Electron想到界面开发的第一印象,肯定是C++ MFC or Qt,无奈C++开发不管是语言还是框架,笔者都感觉太重,效率不高,懒得搞。
然后试了下 Go Walk,中规中矩,文档不太完善,目前GoLang的生态肯定不如Java成熟。如果客户端想加入WebSocket通信,或者Socket通信等其它复杂功能,对笔者而言,还是Java得心应手。
Java JSwing,在功能开发,文件读写,网络通信等,笔者比较顺手,只是Java写界面不是不行,而是界面样式(如圆角,如点击效果,如背景颜色渐变)很多都需要自己去重写 Button Pannel等等组件实现,比较痛苦。好在笔者不想做过于复杂炫酷的交互效果。
Javascript  Electron,Javascript  作为弱语言类型,其实写起来起来还是十分的方便,在样式和交互上,HTML+CSS能够非常方便的去开发定义。功能上,能支持很多系统的本地Api操作,且具有良好的跨平台性。问题就是,正因为简单,所以不想用。还有一点,笔者的插件应用后续可能需要一些其它的协议通信以及office文档,pdf操作功能,不太确定该方案能否完美支持。所以最终:笔者还是选择了Java JSwing方案。

(2)开发应用

程序说明
该程序的运行逻辑为:后端服务器通过WebSocket通知前端网页,前端网页收到WebSocket的消息,并把该消息转成程序参数并Base64加密,同时唤起本地应用程序并将加密字符串当作应用程序的启动参数传入。值得一提的是:Base64加密后的字符串长度+[协议名称]://不能超过2047个字符本地应用程序接收到参数后,进行Base64解密,并开始运行。
一、pom引入依赖
idea新建maven项目并引入依赖
        <!--工具类--><dependency><artifactId>lombok</artifactId><groupId>org.projectlombok</groupId><version>1.18.10</version></dependency><dependency><artifactId>fastjson</artifactId><groupId>com.alibaba</groupId><version>1.2.73</version></dependency><dependency><artifactId>commons-io</artifactId><groupId>commons-io</groupId><version>2.9.0</version></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId><version>3.11</version></dependency><!--工具类--><!-- 日志相关--><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-api</artifactId><version>1.7.25</version></dependency><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.2.3</version></dependency><!--日志相关-->
二、BackgroundPanel类
import java.awt.Graphics;
import java.awt.Image;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import javax.swing.ImageIcon;
import javax.swing.JPanel;
import lombok.extern.slf4j.Slf4j;/**
* 带背景图片的面板
*
* @author LiTing
* @date 2022/1/17 15:16
**/
@Slf4j
public class BackgroundPanel extends JPanel {ImageIcon icon;Image img;public BackgroundPanel(String image) {try {InputStream inputStream = BackgroundPanel.class.getClassLoader().getResourceAsStream(image);if (inputStream != null) {icon = new ImageIcon(inputStream2byte(inputStream));img = icon.getImage();inputStream.close();}} catch (Exception e) {log.warn("初始化背景图片异常:", e);}}@Overridepublic void paintComponent(Graphics g) {super.paintComponent(g);if (img != null) {g.drawImage(img, 0, 0, this.getWidth(), this.getHeight(), this);}}public byte[] inputStream2byte(InputStream inStream) throws IOException {ByteArrayOutputStream swapStream = new ByteArrayOutputStream();byte[] buff = new byte[1024];int rc = 0;while ((rc = inStream.read(buff)) != -1) {swapStream.write(buff, 0, rc);}byte[] in2b = swapStream.toByteArray();return in2b;}
}
三、RadiusButton类
import java.awt.AlphaComposite;
import java.awt.Color;
import java.awt.Font;
import java.awt.GradientPaint;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import java.awt.Shape;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.geom.RoundRectangle2D;
import javax.swing.ImageIcon;
import javax.swing.JButton;/*** 圆角按钮** @author LiTing* @date 2022/1/17 15:18**/
public class RadiusButton extends JButton {/*** 默认颜色 两种渐变色*/private Color BUTTON_COLOR1 = new Color(205, 253, 255);private Color BUTTON_COLOR2 = new Color(47, 108, 154);/*** 按下按钮时字体的默认颜色*/private Color BUTTON_FOREGROUND_COLOR1 = new Color(47, 108, 154);private Color BUTTON_FOREGROUND_COLOR2 = Color.WHITE;/*** 默认字体*/private Font font = new Font("system", Font.PLAIN, 8);/*** 判断是否按下*/private boolean hover;private float clickTran = 0.6F, exitTran = 1F;public RadiusButton(ImageIcon icon) {setIcon(icon);Init();}public RadiusButton(String name) {setText(name);Init();}/*** 修改按下后透明度*/public void setClickTran(float tran) {clickTran = tran;}/*** 修改按下前透明度*/public void setExitTran(float tran) {exitTran = tran;}/*** 上半段渐变颜色*/public void setBUTTON_COLOR1(Color bUTTON_COLOR1) {BUTTON_COLOR1 = bUTTON_COLOR1;}/*** 下半段渐变颜色*/public void setBUTTON_COLOR2(Color bUTTON_COLOR2) {BUTTON_COLOR2 = bUTTON_COLOR2;}/*** 按下前字体颜色*/public void setBUTTON_FOREGROUND_COLOR1(Color bUTTON_FOREGROUND_COLOR1) {BUTTON_FOREGROUND_COLOR1 = bUTTON_FOREGROUND_COLOR1;}/*** 按下后字体颜色*/public void setBUTTON_FOREGROUND_COLOR2(Color bUTTON_FOREGROUND_COLOR2) {BUTTON_FOREGROUND_COLOR2 = bUTTON_FOREGROUND_COLOR2;}public void Init() {setFont(font);setBorderPainted(false);setForeground(BUTTON_FOREGROUND_COLOR1);setFocusPainted(false);setContentAreaFilled(false);addMouseListener(new MouseAdapter() {@Overridepublic void mouseEntered(MouseEvent e) {  //鼠标移动到上面时setForeground(BUTTON_FOREGROUND_COLOR2);hover = true;repaint();}@Overridepublic void mouseExited(MouseEvent e) {  //鼠标移开时setForeground(BUTTON_FOREGROUND_COLOR1);hover = false;repaint();}});}@Overrideprotected void paintComponent(Graphics g) {Graphics2D g2d = (Graphics2D) g.create();int h = getHeight();int w = getWidth();float tran = clickTran;if (!hover) {tran = exitTran;}g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING,RenderingHints.VALUE_ANTIALIAS_ON);GradientPaint p1;GradientPaint p2;if (getModel().isPressed()) {p1 = new GradientPaint(0, 0, new Color(0, 0, 0), 0, h - 1,new Color(100, 100, 100));p2 = new GradientPaint(0, 1, new Color(0, 0, 0, 50), 0, h - 3,new Color(255, 255, 255, 100));} else {p1 = new GradientPaint(0, 0, new Color(100, 100, 100), 0, h - 1,new Color(0, 0, 0));p2 = new GradientPaint(0, 1, new Color(255, 255, 255, 100), 0,h - 3, new Color(0, 0, 0, 50));}g2d.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER,tran));RoundRectangle2D.Float r2d = new RoundRectangle2D.Float(0, 0, w - 1,h - 1, 20, 20);Shape clip = g2d.getClip();g2d.clip(r2d);GradientPaint gp = new GradientPaint(0.0F, 0.0F, BUTTON_COLOR1, 0.0F,h, BUTTON_COLOR2, true);g2d.setPaint(gp);g2d.fillRect(0, 0, w, h);g2d.setClip(clip);g2d.setPaint(p1);g2d.drawRoundRect(0, 0, w - 1, h - 1, 20, 20);g2d.setPaint(p2);g2d.drawRoundRect(1, 1, w - 3, h - 3, 18, 18);g2d.dispose();super.paintComponent(g);}}

TipsMessage类

import lombok.Data;/*** 通知信息对象** @author LiTing* @date 2022/1/17 15:18**/
@Data
public class TipsMessage {private String msgType;private String tips;private String systemName;private long delay = 3000L;
}
NoticeDialogMouseListener类
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;/*** 通知面板的鼠标事件监听器** @author LiTing* @date 2022/1/17 15:16**/
@Data
@Slf4j
public class NoticeDialogMouseListener implements MouseListener {private NoticeDialog noticeDialog;private boolean moveOut;public NoticeDialogMouseListener(NoticeDialog noticeDialog, boolean moveOut) {this.noticeDialog = noticeDialog;this.moveOut = moveOut;}@Overridepublic void mouseClicked(MouseEvent mouseEvent) {}@Overridepublic void mousePressed(MouseEvent mouseEvent) {}@Overridepublic void mouseReleased(MouseEvent mouseEvent) {}@Overridepublic void mouseEntered(MouseEvent mouseEvent) {if (noticeDialog.getTimer() != null) {noticeDialog.getTimer().cancel();noticeDialog.setTimer(null);}}@Overridepublic void mouseExited(MouseEvent mouseEvent) {if (moveOut) {noticeDialog.initTimer();}}
}
ParamValidException类
/*** 参数异常类* @author LiTing* @date 2022/1/17 15:25**/
public class ParamValidException extends Exception {public ParamValidException(String message) {super(message);}
}
NoticeDialog类
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Cursor;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.Font;
import java.awt.Insets;
import java.awt.Toolkit;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import java.util.Timer;
import java.util.TimerTask;
import javax.swing.ImageIcon;
import javax.swing.JDialog;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTextArea;
import javax.swing.border.EmptyBorder;
import lombok.Data;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;/*** 通知对话框** @author LiTing* @date 2022/1/11 14:37**/
@Slf4j
@Data
public class NoticeDialog extends JDialog implements Runnable{/*** icon文件转成byte[]并base64编码*/String iconBase64ByteStr = "iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAAAXNSR0IArs4c6QAAAERlWElmTU0AKgAAAAgAAYdpAAQAAAABAAAAGgAAAAAAA6ABAAMAAAABAAEAAKACAAQAAAABAAAAIKADAAQAAAABAAAAIAAAAACshmLzAAADhElEQVRYCd1WS0hUURj+dEbHx+hMTs6oPUZFQi2LsjCSIIue1iI3lUKBEGlh9FiE0YtoaUUY2CYVImiRCyEXYRS1yHRhmISmNik+xrfOqPN2pnPueK93mLnjvZMgdBZzz5z7///33f985/9P2OmHPR6s4QhfQ2wG+v8iICefk7kpSlJSVy0DaUkK1N1Mh3HaKYmAXJJ1AOMwslZyUIOi/AS0dM3BtLAYwEp46Z8IRMjCcK8kBVv1MQxCffOkMJLAm5AJxEaFo7pcD7XSG6J32IYJk0sARng5JALxMTI8v6yHMlrGRX7zeYqbS5lIJqAi4DUVqYiK9NXvzwGrFFzO1jcKtxx4EikPw9NLm/3AW7vn4XCFVlBFEwgncn90YSO353yKjS0z/L+S5qIJXDmlQ0ZK4CJjGLVLAuUbiyJAq1vBjni+HzefMDlDTj8NsiIB5qwXb2AAF93++/zhu5l5tz5esp4ZvxUJlBVqOdEFOmrtfQugPeDi8UQmoNSfoAQSVXKf1De3m2F3un0wxmad0KgisGeLErQ+SB1BCZQe8f0qLSF07cUAnK5lEhabG2k6BYNbdkIrFV9YA+pYGfIylT4B7xItmCyLOF9lwLu2WUwSAVJZ6LWRjN3eLCVyM7x9wccxyB/BDBQXaPzc4kiKa6+nIzkhArXvJ1BW3c/YyEhTYsftsynQqcULUpZVUPGAdeY/O/st0BBlpy6ll30nJ2BHc9XISYtBh8ECi92Nzn4rmkhGaGNKS4rCsd1qfOk0Y4Fsz0pDkICLtPXWXwugKt+/LQ4UmD8SifDo5aPPaEflmWTQLDS1mfDjjwUHtsfhZN469JAOOToT/IIiuAUsWO+IHaVPDPhttLFL3NPi8H5hOvnq8kId6smNKD9bCfZecOdcCnJSozn7QBNRm2VzenDr5SBuFCVhX3YcF8dK0k8Hv0Ad2qliTsnrj5OgPYJ3YDg//kQUAepA1f64YZTZ18O7VEwM91Jl5HdCj8eDyrohiO0PK24Bny0txDVN4/jWNc8sKyK87tNzyzehqzUDosFpEEkEGFTyU9VgRN+IDYpIrzB7hrz6qHprxPBUcNGxMdhnSARo5u+/GuZKb/+4Hd2DVnxdygwbXMwzJAI0sJWcgE8dcwzGKDmOzxrHxOD52YgWoZ8nWTCTskzHyLQDNgdViPQRcgb4UKGC0xirQoBPRur8L0OUFSVsjXSsAAAAAElFTkSuQmCC";/*** 面板背景图片 需要放classpath下*/String panelBackGround = "panelBackGround.jpg";private int screenWidth;private int screenHeight;private int width = 500;private int height = 300;private int bottomToolKitHeight;private int x;private int y;private int delay;private TipsMessage tipsMessage;private Timer timer = null;private long delaySeconds;public NoticeDialog(TipsMessage tipsMessage) {this.tipsMessage = tipsMessage;}public void init(){NoticeDialogMouseListener noticeDialogMouseListener = new NoticeDialogMouseListener(this,false);setTitle(tipsMessage.getSystemName());ImageIcon icon = new ImageIcon(Base64.getDecoder().decode(iconBase64ByteStr.getBytes(StandardCharsets.UTF_8)));setIconImage(icon.getImage());bottomToolKitHeight = Toolkit.getDefaultToolkit().getScreenInsets(this.getGraphicsConfiguration()).bottom;Dimension dimension = Toolkit.getDefaultToolkit().getScreenSize();screenWidth = dimension.width;screenHeight = dimension.height;//标题文字JLabel titleLabel = new JLabel(tipsMessage.getMsgType());titleLabel.setForeground(Color.BLACK);Font font = new Font("宋体", Font.BOLD | Font.ITALIC, 16);titleLabel.setFont(font);//标题面板JPanel titlePanel = new JPanel(new FlowLayout(FlowLayout.LEFT));titlePanel.setBackground(new Color(4, 97, 142, 40));titlePanel.add(titleLabel);titlePanel.setOpaque(true);titlePanel.addMouseListener(noticeDialogMouseListener);//消息内容区域int maxRowNo = 10;int maxColumnNo = 35;JTextArea msgArea = new JTextArea(tipsMessage.getTips(), maxRowNo, maxColumnNo);msgArea.setBorder(new EmptyBorder(10, 10, 10, 10));msgArea.setLineWrap(true);msgArea.setWrapStyleWord(true);msgArea.setMargin(new Insets(10, 10, 10, 10));JScrollPane jsp = new JScrollPane(msgArea);//消息面板JPanel messagePanel = new JPanel();messagePanel.setBorder(new EmptyBorder(10, 10, 10, 10));messagePanel.setBackground(new Color(98, 195, 243));messagePanel.add(jsp);messagePanel.setOpaque(false);msgArea.addMouseListener(noticeDialogMouseListener);//确定按钮RadiusButton sureButton = new RadiusButton("确定");sureButton.setCursor(new Cursor(12));sureButton.setBackground(new Color(98, 195, 243));sureButton.addActionListener(event -> System.exit(0));sureButton.setBUTTON_COLOR1(new Color(125, 161, 237));sureButton.setBUTTON_COLOR2(new Color(91, 118, 173));sureButton.setBUTTON_FOREGROUND_COLOR1(Color.BLACK);sureButton.setBUTTON_FOREGROUND_COLOR2(Color.GREEN);sureButton.setFont(new Font("宋体", Font.PLAIN, 12));sureButton.setClickTran(0.8F);sureButton.setExitTran(0.3F);sureButton.setSize(new Dimension(60, 40));//确定面板JPanel surePanel = new JPanel();surePanel.setBackground(new Color(98, 195, 243));surePanel.add(sureButton, BorderLayout.SOUTH);surePanel.setPreferredSize(new Dimension(width, 40));surePanel.setOpaque(false);surePanel.addMouseListener(noticeDialogMouseListener);//主面板BackgroundPanel mainPanel = new BackgroundPanel(panelBackGround);mainPanel.setLayout(new BorderLayout());mainPanel.setBorder(new EmptyBorder(0, 0, 20, 0));mainPanel.add(titlePanel, BorderLayout.NORTH);mainPanel.add(messagePanel, BorderLayout.CENTER);mainPanel.add(surePanel, BorderLayout.SOUTH);mainPanel.addMouseListener(new NoticeDialogMouseListener(this,true));//对话框设置x = screenWidth - width;y = screenHeight;this.setLocation(x, y - bottomToolKitHeight - height);this.setSize(width, height);this.getContentPane().add(mainPanel);Toolkit.getDefaultToolkit().beep(); // 播放系统声音,提示一下setAlwaysOnTop(true);setResizable(true);setVisible(true);//延时关闭设置initTimer();}public void initTimer(){if (tipsMessage.getDelay() > 0) {delaySeconds = tipsMessage.getDelay()/1000;this.timer = new Timer();timer.schedule(new TimerTask() {@Overridepublic void run() {System.exit(0);}}, tipsMessage.getDelay());}}@SneakyThrows@Overridepublic void run() {//倒计时设定while (true) {if (tipsMessage.getDelay() > 0 &&timer != null) {setTitle(tipsMessage.getSystemName()+"("+getDelaySeconds()+"s)");delaySeconds--;}Thread.sleep(1000);}}
}
Main启动类
package org.shining;import com.alibaba.fastjson.JSONObject;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import lombok.extern.slf4j.Slf4j;
import org.shining.entity.NoticeDialog;
import org.shining.entity.TipsMessage;
import org.shining.exception.ParamValidException;/*** 程序启动类* @author LiTing* @date 2022/1/17 15:29**/
@Slf4j
public class Main {public static void main(String[] args){TipsMessage tipsMessage;if(args!=null&&args.length>0) {String s = args[0];log.info("传入参数:{}",s);String pt = "yhnotice://";if (s.startsWith(pt)) {s = s.substring(pt.length());}try {tipsMessage = parseParam(s);}catch (ParamValidException paramValidException){tipsMessage = new TipsMessage();tipsMessage.setDelay(-1);tipsMessage.setSystemName("盈狐制单系统");tipsMessage.setMsgType("非法的传入参数");tipsMessage.setTips("参数:"+s);}}else{log.info("没有参数传入");tipsMessage = new TipsMessage();tipsMessage.setDelay(3000L);tipsMessage.setSystemName("盈狐制单系统");tipsMessage.setMsgType("运行成功通知");tipsMessage.setTips("程序启动成功\n请打开制单系统网页版");}NoticeDialog noticeDialog = new NoticeDialog(tipsMessage);noticeDialog.init();noticeDialog.run();}private static TipsMessage parseParam(String s) throws ParamValidException {TipsMessage tipsMessage;try {int i = s.length()%4;s = s.substring(0,s.length()-i);s = new String(Base64.getDecoder().decode(s.getBytes(StandardCharsets.UTF_8)), StandardCharsets.UTF_8);tipsMessage = JSONObject.parseObject(s, TipsMessage.class);if (tipsMessage.getDelay() == 0) {tipsMessage.setDelay(3000L);}return tipsMessage;}catch (Exception e){throw new ParamValidException("传入参数不合法");}}
}

(3)测试效果

运行Main类的main方法

(4)开始打包

这里的打包必须把各种第三方依赖包也打进去形成可独立运行的Jar包。所以,pom文件写入以下内容
   <build><resources><resource><directory>src/main/java</directory><!--所在的目录--><includes><!--包括目录下的.properties,.xml文件都会扫描到--><include>**/*.xml</include></includes><filtering>false</filtering></resource><resource><directory>src/main/resources</directory><filtering>true</filtering></resource></resources><plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><version>3.1</version><configuration><source>${java.version}</source><target>${java.version}</target></configuration></plugin><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-shade-plugin</artifactId><version>2.3</version><executions><execution><phase>package</phase><goals><goal>shade</goal></goals><configuration><transformers><transformerimplementation="org.apache.maven.plugins.shade.resource.ManifestResourceTransformer"><!-- 这里要替换成你自己的启动类--><mainClass>org.shining.Main</mainClass></transformer></transformers></configuration></execution></executions></plugin></plugins></build>
运行 mvn clean package

(5)转成exe文件

找到exe4j破解版,安装运行并change License.(否则转出来的exe会有exe4j的弹窗)

 直接下一步,选择"JAR in EXE"  mode

 下一步,填入exe的名称以及exe文件的保存目录

下一步,勾选Allow -console parameter,再次输入应用名称,点击高级选项,选择32-bit or 64-bit

下一步

 再下一步,勾选Aways

 下一步:VmOption输入 -Dfile.encoding=utf-8,选择Jar包和启动类


下一步,选择 输入jdk版本,且选择高级选项


然后一直下一步直至完成

点击Save As 将此次配置保存至某个文件夹,防止下次代码变动,还得重头配一遍。
有此次配置,只要相关文件目录不变动,下次输出exe,只需要加载配置文件,一直下一步至完成即可。
保存完成,点击exit退出即可。
找到exe文件运行一次


(6)封装安装程序

 安装并运行innoSetup程序;新建空白文件;写入以下内容:
; 脚本由 Inno Setup 脚本向导 生成!
; 有关创建 Inno Setup 脚本文件的详细资料请查阅帮助文档!
; 替换成自己的安装后的程序名称  禁止中文
#define MyAppName "yhnotice"
#define MyAppVersion "1.1.0.0"
#define MyAppPublisher "浙江LJKDSHAKJ有限公司"
#define MyAppURL "http://www.aaabbb.com"
;替换成自己的安装后的程序名称  禁止中文
#define MyAppExeName "yhnotice.exe"
#define MyJreName "jre"
;替换成自己的协议名称
#define MyProName "yhnotice"
[Setup]
; 注: AppId的值为单独标识该应用程序。
; 不要为其他安装程序使用相同的AppId值。
; (若要生成新的 GUID,可在菜单中点击 "工具|生成 GUID"。)
AppId={{13B3A4B2-B9A4-474B-B068-BA6CF2A00592}
AppName={#MyAppName}
AppVersion={#MyAppVersion}
;AppVerName={#MyAppName} {#MyAppVersion}
AppPublisher={#MyAppPublisher}
AppPublisherURL={#MyAppURL}
AppSupportURL={#MyAppURL}
AppUpdatesURL={#MyAppURL}
DefaultDirName={autopf}\{#MyAppName}
DisableProgramGroupPage=yes
;替换成自己的安装程序输出目录
OutputDir=D:\opt
;替换成自己的安装程序名称
OutputBaseFilename=setup
Compression=lzma
SolidCompression=yes
WizardStyle=modern
AlwaysRestart=yes
[Languages]
Name: "chinesesimp"; MessagesFile: "compiler:Default.isl"[Tasks]
Name: "desktopicon"; Description: "{cm:CreateDesktopIcon}"; GroupDescription: "{cm:AdditionalIcons}"; Flags: unchecked[Files]
;替换成自己的exe4j输出exe文件输出路径
Source: "D:\opt\yhnotice.exe"; DestDir: "{app}"; Flags: ignoreversion
;替换成自己的jre路径
Source: "C:\Program Files\Java\jre1.8.0_144\*"; DestDir: "{app}\{#MyJreName}"; Flags: ignoreversion recursesubdirs createallsubdirs[Registry]
;写入自定义协议注册表
Root: HKLM; Subkey: "SOFTWARE\Microsoft\Windows\CurrentVersion\Run"; ValueType: string; ValueName: "{#MyProName}"; ValueData: "{app}\{#MyAppExeName}"
Root: HKCR; SubKey: {#MyProName}; ValueName:"URL Protocol";ValueData: ""; ValueType: string; Flags: CreateValueIfDoesntExist UninsDeleteKey;
Root: HKCR; SubKey: {#MyProName}; ValueData: "URL:yhnotice Protocol Handler"; ValueType: string; Flags: CreateValueIfDoesntExist UninsDeleteKey;
Root: HKCR; SubKey: {#MyProName}\DefaultIcon; ValueData: "{app}\{#MyAppExeName}"; ValueType: string; Flags: CreateValueIfDoesntExist UninsDeleteKey;
Root: HKCR; SubKey: {#MyProName}\shell\open\command; ValueData: """{app}\{#MyAppExeName}"" ""%1"""; Flags: CreateValueIfDoesntExist UninsDeleteKey; ValueType: string;
Root: HKLM; Subkey: "Software\Policies\Google\Chrome"; ValueType: qword; ValueName: "ExternalProtocolDialogShowAlwaysOpenCheckbox";Flags:CreateValueIfDoesntExist UninsDeleteKey;ValueData: "1";
;替换成自己的域名  ps:""内的"用""进行转义 {用{{进行转义
Root: HKLM; Subkey: "Software\Policies\Google\Chrome"; ValueType: string; ValueName: "AutoLaunchProtocolsFromOrigins";Flags:CreateValueIfDoesntExist UninsDeleteKey;ValueData: "[{{""protocol"":""{#MyProName}"",""allowed_origins"":[""localhost"",""http://localhost:8080"",""http://make.frp.aaabbb.com"",""http://make.test.inner.aaabbb.com"",""http://make.test.out.aaabbb.com"",""http://make.online.aaabbb.com"",""https://localhost:8080"",""https://make.frp.aaabbb.com"",""https://make.test.inner.aaabbb.com"",""https://make.test.out.aaabbb.com"",""https://make.online.aaabbb.com""]}]";
[Icons]
Name: "{autoprograms}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"
Name: "{autodesktop}\{#MyAppName}"; Filename: "{app}\{#MyAppExeName}"; Tasks: desktopicon[Run]
Filename: "{app}\{#MyAppExeName}"; Description: "{cm:LaunchProgram,{#StringChange(MyAppName, '&', '&&')}}"; Flags: nowait postinstall skipifsilent

点击Compile,找到输出目录的setup.exe程序。右键以管理员的身份安装并重启电脑。

(7)测试能否浏览器唤起

新建网页test.html
写入以下内容:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Title</title>
</head>
<body>
<h1>bbbbbbbbbbbbbb</h1>
<br><a href="yhnotice://eyJkZWxheSI6MzAwMCwibXNnVHlwZSI6IumCruS7tua0vuWNlemAmuefpSIsInN5c3RlbU5hbWUiOiLliLbljZXns7vnu58iLCJ0aXBzIjoi6YKu5Lu25rWB5rC05Y+377yaMjAyMTMzMDkxMTAxMDAwMVxu5Y+R5Y2V5a6i5oi377yaIOS4iua1t+WNjum4o+aKpeWFs+i9r+S7tuaciemZkOWFrOWPuFxu5Y+R5Lu25Lq677ya5Y2i5pmT5pyIKDM4Mjk4MzIzODkyQHFxLmNvbSlcbumCruS7tuagh+mimO+8mua0i+WxseS4pOaLvOetiemAmuefpeaKpeWFsyAx5pyIMTPml6XoiLnmnJ8g5oql5YWz6LWE5paZXG7mlLbku7bnrrHvvJo3NzM3NzgzMjc4QHFxLmNvbVxuIn0=">cccccccccccccccccccc</a>
</body>
</html>
右键test.html选择浏览器打开

点击ccccccccc,这里因为是File访问,所以有弹窗和需要勾选。
如果是通过我们innoSetup文件中配置的域名来访问,那么不需要勾选,会直接弹出。


结尾

说明一
如此,一个可被网页唤起的的插件程序开发完成。
目前还残留1个问题:程序唤起时目前有冷却时间,几秒被不能被重复唤起。除非在桌面google浏览器的图标上,右键 属性 快捷方式  目标 后面 加上参数 --autoplay-policy=no-user-gesture-required

说明二

主要也需要了解一些windows系统的注册表知识
1.Windows注册表(如自定义协议)
2.Google浏览器的注册表支持文档,需要翻墙(https://admx.help/?Category=ChromeEnterprise&Policy=Google.Policies.Chrome::AutoplayAllowed),通过修改注册表来绕过google浏览器对我们程序的安全限制。
3. 如果需要微软的edg浏览器支持,那么也可以了解edg浏览器的注册表支持文档。(https://admx.help/?Category=EdgeChromium)

说明三

如何查看windows系统注册表

输入regedit可以打开注册表
例如: 系统自启动:SOFTWARE\Microsoft\Windows\CurrentVersion\Run 建立一个字符串项,项名随便起,字符串值 填入自己的程序安装目录。那么windows系统启动时,会自动启动。
说明四
我们这里是把写入注册表的过程,封装近了到了程序的安装脚本中。
其实也可以或者后续有变动,可以通过自定义一个.reg为后缀的文件进行注册表的操作与修改。这种网上比较多的案例。
说明五
如果网页是http访问,那么网页唤起本地应用程序时,不会出现记住我的选项。所以需要在注册表写入:
Software\Policies\Google\Chrome
AutoLaunchProtocolsFromOrigins = [{"protocol":"yhnotice","allowed_origins":["localhost","http://localhost:8080","http://make.frp.aaabbb.com","http://make.test.inner.aaabbb.com","http://make.test.out.aaabbb.com","http://make.online.aaabbb.com"]}]
ExternalProtocolDialogShowAlwaysOpenCheckbox = 1其中:AutoLaunchProtocolsFromOrigins 类型为字符串  ExternalProtocolDialogShowAlwaysOpenCheckbox 类型为数值类型。数值类型 32位系统用Dword,64位系统用Qword

chorme唤起Java开发的本地程序全采坑记相关推荐

  1. java linux 管理系统_用Java开发一个本地服务管理软件

    使用Java开发一个本机服务管理程序,能够控制本机Tomcat.Apache服务的开启和关闭,图形界面控制.用户可以自己扩展其他服务,用来学习图形界面编程.多线程.事件响应等都不错. 一.最终界面 先 ...

  2. java驱动pl sql优点_用PL/SQL和Java开发Oracle8i应用程序

    用PL/SQL和Java开发Oracle8 i应用程序 随着Oracle8i的发布,Oracle 在数据库里支持了二种主要的编程语言??PL/SQL和Java.今天,Oracle的许多客户既使用PL/ ...

  3. 关于java开发邮件接收程序的一点总结

    用java开发邮件接收程序需要用到javamail包,资源地址:http://java.sun.com/products/javamail/,最新版是 JavaMail 1.4.3 ,下载地址是:ht ...

  4. Java开发环境搭建超全详解

    在项目产品开发中,开发环境搭建是软件开发的首要阶段,也是必须阶段,只有开发环境搭建好了,方可进行开发,良好的开发环境搭建,为后续的开发工作带来极大便利. 对于大公司来说,软件开发环境搭建工作一般是由运 ...

  5. 指定应用程序网络连接_总结Java开发Web应用程序应该理解的几个知识点

    前言 前面我们对Web应用开发的底层技术做了一些串联,也就是从应用程序的本质出发来理解为什么我们的应用程序架构的演变. 特别是Spring框架的出现,它在Web应用开发中扮演的角色,特别是Servle ...

  6. java开发中对于程序员的几点建议,你们有想到吗?

    建议一:只有真正喜欢才能写好程序 喜欢写程序,做程序员就是上天堂: 不喜欢写程序,做程序员就是下地狱: 程序员需要整天趴在电脑前,经常没日没夜的,非常辛苦,而且工作来不得半点虚假,少写一个标点符号都不 ...

  7. erp开发和java开发区别_Java程序员求职必学:Spring boot学习指南!

    黑马程序员上海中心 学姐微信:CZBKSH 关注 咳咳,今天学姐就来和你们说说Spring对于Java程序员的重要性. 首先,Spring 官网首页是这么介绍自己的--"Spring: th ...

  8. java web底层原理_详解Java开发Web应用程序的底层原理

    前言 前面一篇文章,我从整个应用程序的整体以及跟运行环境的关系简单聊了一下我们现在常用的Spring框架的设计基础和准则,其中主要是控制反转和依赖注入,以及容器化编程等概念. 这里我不想去复述这些概念 ...

  9. Java开发微信小程序授权登录

    最近对接开发微信小程序,需要获取用户的openid使用支付,所以记下这篇通用小程序授权笔记. 这里使用到开源工具Wx-Java 此致 致敬 binarywang 大佬 maven引入如下 <!- ...

  10. Taro小程序采坑记

    Taro,京东凹凸实验室出品的适配多端的一个框架, Taro 是一套遵循 React 语法规范的 多端开发 解决方案.现如今市面上端的形态多种多样,Web.React-Native.微信小程序等各种端 ...

最新文章

  1. 001-Spring Cloud Edgware.SR3 升级最新 Finchley.SR1,spring boot 1.5.9.RELEASE 升级2.0.4.RELEASE注意问题点...
  2. NOIP 2012 同余方程
  3. 如何快速直接从Web of Science下载文献,保存到Endnote
  4. 真香!spaCy+Cython比Python快100倍.....
  5. Java进击C#——应用开发之Asp.net MVC
  6. idea创建web项目的几种方式(没内容,自用)
  7. Python之网络编程(粘包、粘包解决方案)
  8. Android Error:Could not find lottie.jar
  9. vc 编译器的一些精典报错
  10. 【以太坊开发】发币指南--进阶篇
  11. Jason与Xml的解析过程
  12. android 字体像素转换工具类_Android点9图机制及在聊天气泡中的应用
  13. 微服务架构实践之邮件通知系统改造
  14. 专访尹毅(法师Seay):少年成名,野路子的奇妙逆袭 - FreeBuf.COM | 关注黑客与极客...
  15. 转自瑞星:“永恒之蓝”WannaCry勒索病毒分析报告
  16. linux zip文件解压命令详解
  17. C语言求若干个数的均值和方差
  18. 武汉大学计算机学院保研清华,清北11人保送武大法学院,网友表示看不懂,高校老师表示:正常...
  19. 我私藏的那些实用的终端命令行工具
  20. 万字长文!推荐一款日志切割神器

热门文章

  1. 比Kafka Mangaer更优秀的开源监控工具-Kafka Eagle
  2. 中国标准时间、标准时间、时间戳时间格式转换
  3. 英:最常用有用的英语口语900句
  4. cpci无法检索_CPCI是什么检索
  5. [《所遇随心》偶感小记]2012年8月28日
  6. Istio入坑指南(二) Istio的安装与简单的使用
  7. 把redis部署到百度BAE上时的注意点
  8. 将支付网关添加到您的WooCommerce商店
  9. llS 10.0详细错误-404.0 - NotFound
  10. python pdf文件处理