Electron 读取本地配置 增加缩放功能(ctrl+scroll)

news2025/1/12 0:54:29

最近,一个之前做的electron桌面应用,需要增加两个功能;第一是读取本地的配置文件,然后记载配置文件中的ip地址;第二就是增加缩放功能;

第一,配置本地文件

首先需要在vue工程根目录中,新建一个config.json文件;如下图

config.json内容如下:

{
  "ip": "1.11.21.219",
  "port": 30002,
}

然后在vue.config.js中需要排除这个文件,如下:

    // 添加electron - app -icon
    pluginOptions: {
        electronBuilder: {
            builderOptions: {
                productName: 'xxxx', //项目名,也是生成的安装文件名
                //copyright: "Copyright © 2019",//版权信息
                win: {
                    icon: './public/favicon.ico',
                    // 以管理员权限运行
                    requestedExecutionLevel: 'requireAdministrator',
                    target: [{
                        target: "nsis", //利用nsis制作安装程序
                        arch: [
                            "x64", //64位
                        ]
                    }],
                    
                },
                nsis: {
                    oneClick: false, // 是否一键安装
                    allowElevation: true, // 允许请求提升。 如果为false,则用户必须使用提升的权限重新启动安装程序。
                    allowToChangeInstallationDirectory: true, // 允许修改安装目录
                    installerIcon: "./public/favicon.ico", // 安装图标
                    uninstallerIcon: "./public/favicon.ico", //卸载图标
                    installerHeaderIcon: "./public/favicon.ico", // 安装时头部图标
                    createDesktopShortcut: true, // 创建桌面图标
                    createStartMenuShortcut: true, // 创建开始菜单图标
                    shortcutName: "MIES", // 图标名称
                },
                directories: {
                    output: "./MIES_SETUP" //输出文件路径
                },



                /**** 注意这里 配置config.json ****/





                extraResources: [
                    { "from": "./config.json", "to": "../" }
                ],
            },
            nodeIntegration: true,
            preload: 'src/preload.js'
        }
    },
    // 

然后,安装桌面应用之后,会在安装目录出现这个配置好的config.json.

第二,读取本地配置文件,创建window

在backgroundjs中,使用nodejs的fs模块读取根目录下的config.json文件,动态获取配置的ip和端口,然后创建window。

代码如下:

............







// Create the browser window.
let win = null;

async function createWindow() {
  // 读取信息
  var exePath = path.dirname(app.getPath("exe")).replace(/\\/g, "/");
  // console.log(exePath);
  var configPath = `${exePath}/config.json`;

  var sockets = [];


   // ********************   这里是主要功能   ************************* //
   // 读取本地文件 获取配置信息

  fs.readFile(configPath, "utf-8", async (err, data) => {
    if (data) {
      const cp = require("child_process");
      let res = JSON.parse(data);
      const PAGE_URL = `https://${res.ip}:${res.port}`;
      // 创建窗口
      win = new BrowserWindow({
        width: 800,
        height: 600,
        webPreferences: {
          // Use pluginOptions.nodeIntegration, leave this alone
          // See nklayman.github.io/vue-cli-plugin-electron-builder/guide/security.html#node-integration for more info
          // process.env.ELECTRON_NODE_INTEGRATION
          nodeIntegration: process.env.ELECTRON_NODE_INTEGRATION,
          contextIsolation: !process.env.ELECTRON_NODE_INTEGRATION,
          preload: path.join(__dirname, "/preload.js"),
          webSecurity: false,
          allowRunningInsecureContent: false,
          //zoomFactor: 0.6,
        },
        icon: path.join(__dirname, "./favicon.ico"),
      });

      if (process.env.WEBPACK_DEV_SERVER_URL) {
        // Load the url of the dev server if in development mode
        //await win.loadURL(process.env.WEBPACK_DEV_SERVER_URL);
        console.log("PAGE_URL ----- ", PAGE_URL);
        await win.loadURL(PAGE_URL);
        if (!process.env.IS_TEST) win.webContents.openDevTools();
      } else {
        createProtocol("app");
        // Load the index.html when not in development
        // win.loadURL('app://./index.html');
        //win.loadURL(path.join(__dirname, "./index.html"));
        // https://www.electronjs.org
        console.log("PAGE_URL ----- ", PAGE_URL);
        win.loadURL(PAGE_URL);
      }

      //进入软件即开启全屏
      //win.setFullScreen(true);

      // 最大化
      win.maximize();

      //配置ESC键退出全屏 ESC
      globalShortcut.register("Alt+CommandOrControl+Q", () => {
        win.setFullScreen(false);
      });

      // 配置设置快捷键
      globalShortcut.register("Alt+CommandOrControl+S", () => {
        win.webContents.send("asynchronous-message", "123");
      });

      // ctrl_alt_f 打开全屏设置
      globalShortcut.register("Alt+CommandOrControl+F", () => {
        win.setFullScreen(true);
      });

      // 主进程缩小窗口
      ipcMain.on("window-min", function() {
        // 收到渲染进程的窗口最小化操作的通知,并调用窗口最小化函数,执行该操作
        win.minimize();
      });

      // ctrl_alt_i 手动打开开发者工具
      globalShortcut.register("Alt+CommandOrControl+I", () => {
        win.webContents.openDevTools({
          mode: "bottom",
        });
      });

      // 获取安装地址
      // globalShortcut.register('Alt+CommandOrControl+P', () => {
      //     win.webContents.send("asynchronous-message", path.dirname(app.getPath('exe')));
      // })

      // 手动打开开发者工具
      // ipcMain.on('open-dev', function () { // 收到渲染进程的窗口最小化操作的通知,并调用窗口最小化函数,执行该操作
      //     win.webContents.openDevTools({
      //         mode: 'bottom'
      //     })
      // })

      // 设置顶部菜单
      // 自定义一些菜单
      const appMenuTemplate = [
        {
          label: "窗口",
          submenu: [
            {
              label: "打开全屏",
              click: () => {
                win.setFullScreen(true);
              },
            },
            {
              label: "退出全屏",
              click: () => {
                win.setFullScreen(false);
              },
            },
          ],
        },
        {
          label: "设置",
          submenu: [
            // {
            //   label: "设置首页",
            //   click: () => {
            //     win.webContents.send("asynchronous-message", "123");
            //   },
            // },
            {
              label: "强制刷新",
              role: "forceReload",
            },
            {
              label: "退出",
              role: "quit",
            },
            {
              label: "开发者选项",
              click: () => {
                win.webContents.openDevTools({
                  mode: "bottom",
                });
              },
            },
          ],
        },
      ];

      const menu = Menu.buildFromTemplate(appMenuTemplate);
      Menu.setApplicationMenu(menu);

      // 窗口关闭的监听
      win.on("closed", (event) => {
        win = null;
      });

      // 点击关闭
      win.on("close", (event) => {
        const closeWinFlagValue = dialog.showMessageBoxSync(win, {
          type: "info",
          buttons: ["最小化到托盘", "直接退出", "取消"],
          title: "提示",
          message: "确定要退出吗?",
          defaultId: 0,
          cancelId: 2,
        });
        console.log("closeWinFlagValue", closeWinFlagValue);
        event.preventDefault();
        if (closeWinFlagValue === 0) {
          // 托盘对象
          var appTray = null;
          // 系统托盘右键菜单
          var trayMenuTemplate = [
            {
              label: "显示",
              click: function() {
                !win.isVisible() ? win.show() : null;
              },
            },
            {
              label: "退出",
              click: function() {
                app.quit();
              },
            },
          ];
          win.hide();
          // 系统托盘图标目录 path.join(__static, './logo_1.ico')
          let trayIcon = path.join(__dirname, "./favicon.ico");

          appTray = new Tray(trayIcon);

          // 图标的上下文菜单
          const contextMenu = Menu.buildFromTemplate(trayMenuTemplate);

          // 设置此托盘图标的悬停提示内容
          appTray.setToolTip("上海局高铁机务管理智能评价系统");

          // 设置此图标的上下文菜单
          appTray.setContextMenu(contextMenu);

          // 单击击托盘显示隐藏
          appTray.on("click", () => {
            win.isVisible() ? win.hide() : win.show();
            // 关闭托盘显示
            appTray.destroy();
            appTray = null;
          });
        } else if (closeWinFlagValue === 1) {
          app.exit();
        } else if (closeWinFlagValue === 2) {
          win.focus();
          win.show();
        }
      });

      // 设置托盘
      // Exit cleanly on request from parent process in development mode.

      if (process.platform === "win32") {
        /* process.on('message', (data) => {
            if (data === 'graceful-exit') {
                app.quit()
            }
        }) */
      } else {
        process.on("SIGTERM", () => {
          app.quit();
        });
      }

      // D:/dev_tools/MIES
      //console.log('读取本地文件 == ', res);
      // let socketPath = exePath + "/WebMiddleware.exe";
      // //let child = cp.spawn(socketPath, [res.ip, res.port])
      // cp.exec(`${socketPath} ${res.ip} ${res.port}`, (err, stdout, stderr) => {
      //   console.log("err, stdout, stderr", err, stdout, stderr);
      // });

      // 向vue发送配置的wsip和端口
      let ws_path = `wss://${res.ws_ip}:${res.ws_port}`;
      const http = require("http"); // 创建服务器对象
      const server = http.createServer();
      const closeServer = () => {
        sockets.forEach(function(socket) {
          socket.destroy();
        });
        server.close(function() {
          console.log("close server!");
        });
      };
      server.listen(res.local_port); // 对错误进行捕获
      server.on("error", (err) => {
        if (err.code == "EADDRINUSE") {
          // 如果目标端口被占用将使用
          // NodeJS 随机分配的端口号
          server.listen(0);
        }
      }); // 在成功监听后,向终端输出被监听的端口号
      server.on("listening", () => {
        console.log(
          "【HTTP Server is running at http://127.0.0.1:" +
            server.address().port +
            " 】"
        );
      });
      server.on("connection", function(socket) {
        sockets.push(socket);
        //console.log('sockets', sockets);
        socket.once("close", function() {
          sockets.splice(sockets.indexOf(socket), 1);
        });
      });
      server.on("request", function(req, res) {
        const url = req.url;
        if (url === "/getWsPath") {
          res.setHeader("content-type", "application/json");
          res.end(ws_path);
          closeServer();
        } else {
          res.writeHeader(404);
          res.end("404 not found");
          closeServer();
        }
      });
    }
  });
}

至此就可以完成读取本地文件,获取配置信息功能。

第三,完成缩放功能

需求要求实现和浏览器一样,ctrl加上鼠标滚轮,可以完成页面的缩放,具体代码如下:

let level = 0;
      // 缩放
      win.webContents.on('zoom-changed',(e, zoomDirection)=>{
        if (zoomDirection === 'in') {
          level = level >= 3 ? level : level += 0.2
        } else {
          level = level <= -3 ? level : level -= 0.2
        }
        win.webContents.setZoomLevel(level)
      })

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

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

相关文章

华为OD机试 - 芯片资源限制(Java 2024 C卷 100分)

华为OD机试 2024C卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;A卷B卷C卷&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;每一题都有详细的答题思路、详细的代码注释、样例测试…

uniapp微信小程序消息订阅详解

一、微信公众平台申请订阅模板 注意&#xff1a;订阅信息 这个事件 是 当用户 点击的时候触发 或者 是 支付成功后触发&#xff0c; 用户勾选 “总是保持以上选择&#xff0c;不再询问” 之后或长期订阅&#xff0c;下次订阅调用 wx.requestSubscribeMessage 不会弹窗&#xf…

爬虫的验证码处理

1.我们先进入chrome浏览器的审查页面找到input方法&#xff1a; 为了不少找到一个input&#xff0c;我们ctrlf的方法输入input来查找 看见我们有6个需要输入的参数。 除了上面几个的input参数&#xff0c;我们还需要获取验证码的图片&#xff0c;后续要将字母填入进去。 二.安…

【蓝桥杯】矩阵快速幂

一.快速幂概述 1.引例 1&#xff09;题目描述&#xff1a; 求A^B的最后三位数表示的整数&#xff0c;A^B表示&#xff1a;A的B次方。 2&#xff09;思路&#xff1a; 一般的思路是&#xff1a;求出A的B次幂&#xff0c;再取结果的最后三位数。但是由于计算机能够表示的数字…

Vue ElementPlus Form、Form-item 表单

Form 表单 由输入框、选择器、单选框、多选框等控件组成&#xff0c;用以收集、校验、提交数据&#xff0c;组件升级采用了 flex 布局&#xff0c;以替代旧版本的 float 布局。 典型表单 包括各种表单项&#xff0c;比如输入框、选择器、开关、单选框、多选框等。 在 Form 组件…

数据结构之单链表实现(JAVA语言+C语言)

一、理论 1 单链表结构 2 增、删、查 、改思路 &#xff08;增&#xff09;直接添加放到最后即可。按顺序添加&#xff1a;找到要修改的节点的前一个节点&#xff0c;插入新节点&#xff08;&#xff09;。&#xff08;改&#xff09;要修改的节点修改内容即可。&#xff08;…

03-MySQl数据库的-用户管理

一、创建新用户 mysql> create user xjzw10.0.0.% identified by 1; Query OK, 0 rows affected (0.01 sec) 二、查看当前数据库正在登录的用户 mysql> select user(); ---------------- | user() | ---------------- | rootlocalhost | ---------------- 1 row …

Docker:使用MinIO搭建对象存储平台

请关注微信公众号&#xff1a;拾荒的小海螺 1、简述 MinIO是一个基于对象存储技术的开源项目&#xff0c;它可以帮助用户快速搭建起私有的、高性能的对象存储平台。MinIO兼容Amazon S3 API&#xff0c;使得用户可以使用标准的S3工具和SDK来访问和管理MinIO存储的数据。此外&a…

查找--二分查找(Binary Search)

二分查找属于静态查找表&#xff0c;当以有序表表示静态查找表时&#xff0c;查找函数可用折半查找来实现。 查找过程&#xff1a;先确定待查记录所在的范围&#xff08;区间&#xff09;&#xff0c;然后逐步缩小范围直到找到或找不到该记录为止。 以处于区间中间位置记录的…

B样条曲线(记录)

B样条曲线的生成靠的两点&#xff1a; 1、控制点 2、基函数 B样条曲线的基函数是一个De Boor的递归表达式[1]&#xff1a; (1) (2) 其中是第个阶基函数。 而B样条曲线可以表示为[2]&#xff1a; (3) 如何理解上式&#xff1f;首先&#xff0c;我们知道&#xff0c;如果一个函数…

高端的电子画册,手机打开你见过吗?

手机阅读的高端电子画册&#xff0c;你见过吗&#xff1f;随着移动互联网的发展&#xff0c;越来越多的人选择在手机上阅读电子画册&#xff0c;而不是传统的纸质画册。这种趋势不仅节省了纸张资源&#xff0c;还提升了阅读体验。用户可以通过触摸屏幕、放大缩小、翻页等操作与…

【Blockchain】区块链浏览器 | 以太坊Etherscan比特币Blockchain门罗币Monero

区块链浏览器概述 区块链浏览器是一种软件,它使用API(应用程序编程接口)和区块链节点从区块链中提取各种数据&#xff0c;然后使用数据库来排列搜索到的数据&#xff0c;并以可搜索的格式将数据呈现给用户。 用户的输入是资源管理器上的可搜索项&#xff0c;然后通过数据库上…

empdll文件安装在哪里,详细的修复教程分享

在我们运行《荒野大镖客2》游戏的时候&#xff0c;有些玩家在游玩过程中可能会遇到emp.dll文件丢失的问题。此文件作为游戏运行过程中不可或缺的动态链接库&#xff08;DLL&#xff09;组件之一&#xff0c;丢失会导致游戏无法正常运行。小编将介绍5种解决emp.dll文件丢失的方法…

linux安装Zookeeper的详细步骤

1.Java环境确认 确保已经安装了Java环境&#xff0c;没有的自行安装 2.官网下载包 Apache ZooKeeper 3.安装 3.1上传到linux&#xff0c;解压 我的目录为/root/apache-zookeeper-3.8.4-bin 进入到/root/apache-zookeeper-3.8.4-bin/conf目录下&#xff0c;执行命令复制zoo…

C++2D原创我的世界1.00.3版本上市!!!

我很郁闷&#xff0c;为什么就是整不了昼夜交替啊喂&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01;&#xff01; 虽然这看上去很简单&#xff0c;但做起来要我命&#xff01;&#xff01;&#xff01; 优化过后总共1312行&#xff0c…

微信公众号迁移公证书在哪?

公众号迁移有什么作用&#xff1f;只能变更主体吗&#xff1f;很多小伙伴想做公众号迁移&#xff0c;但是不知道公众号迁移有什么作用&#xff0c;今天跟大家具体讲解一下。首先公众号迁移最主要的就是修改公众号的主体了&#xff0c;比如我们公众号原来是A公司的&#xff0c;现…

指针强化练习(详解)

更多学习内容 结构体内存对齐 和 位段-CSDN博客指针初级&#xff08;基础知识&#xff09;-CSDN博客指针进阶(深入理解)-CSDN博客 目录 1.sizeof与strlen的区别 2.一维数组 3.字符指针 4.二维数组 5.指针运算(笔试题) 6.函数指针 1.sizeof与strlen的区别 请思考以下运行结…

第1章.提示词:开启AI智慧之门的钥匙

什么是提示词&#xff1f; 提示词&#xff0c;是引导语言模型的指令&#xff0c;让用户能够驾驭模型的输出&#xff0c;确保生成的文本符合需求。 ChatGPT&#xff0c;这位文字界的艺术大师&#xff0c;以transformer架构为基石&#xff0c;能轻松驾驭海量数据&#xff0c;编织…

Chrome浏览器隐藏的截图功能配置及使用

来自实用又方便&#xff0c;轻松打开Chrome浏览器隐藏的截图功能&#xff01;​​​​​​​ 一、通过谷歌Chrome浏览器 现在直接通过谷歌Chrome浏览器内置功能&#xff0c;免安装扩充插件也可以实现Chrome的截图和长截图功能了&#xff01; 也不需要额外安装任何截图工具 &a…

【C++航海王:追寻罗杰的编程之路】priority_queue(优先队列) | 容器适配器你知道哪些?

目录 1 -> priority_queue的介绍和使用 1.1 -> priority_queue的介绍 1.2 -> priority_queue的使用 1.3 -> priority_queue的模拟实现 2 -> 容器适配器 2.1 -> 什么是适配器 2.2 -> STL标准库中stack和queue的底层结构 2.3 -> deque的介绍 2.…