主頁 > 企業開發 > 基于electron25+vite4創建多視窗|vue3+electron25新開模態表單

基于electron25+vite4創建多視窗|vue3+electron25新開模態表單

2023-05-31 08:42:23 企業開發

在寫這篇文章的時候,查看了下electron最新穩定版本由幾天前24.4.0升級到了25了,不得不說electron團隊迭代速度之快!

前幾天有分享一篇electron24整合vite4全家桶技術構建桌面端vue3應用示例程式,

https://www.cnblogs.com/xiaoyan2017/p/17436076.html

這次繼續接著上次專案,主要介紹electron25結合vue3技術實作創建多開視窗及視窗間主/渲染行程通信知識,

隨著electron快速更新,結合vite的高效構建運行速度,現在新開一個獨立視窗,打開速度極快,

electron官網主行程模塊BrowserWindow用于創建一個新視窗的方法,提供了非常豐富的API操作用法,

https://www.electronjs.org/docs/latest/api/browser-window

// In the main process.
const { BrowserWindow } = require('electron')

const win = new BrowserWindow({ width: 800, height: 600 })

// Load a remote URL
win.loadURL('https://github.com')

// Or load a local HTML file
win.loadFile('index.html')

如果每次都new一個BrowserWindow視窗,顯得有些笨拙且復雜,今天要分享的是封裝BrowserWindow方法,只需傳入配置引數,即可快速生成一個獨立視窗,

createWin({
    title: '關于About.vue',
    route: '/about',
    width: 600,
    height: 400,
    background: '#fafffa',
    resize: true
})

新建一個windows/index.js檔案,

/**
 * 封裝多視窗管理器
 * @author YXY
 */

const { app, BrowserWindow, ipcMain } = require('electron')
const { join } = require('path')

process.env.ROOT = join(__dirname, '../../')

const isDevelopment = process.env.NODE_ENV == 'development'
// const winURL = isDevelopment ? 'http://localhost:3000/' : join(__dirname, 'dist/index.html')
const winURL = isDevelopment ? process.env.VITE_DEV_SERVER_URL : join(process.env.ROOT, 'dist/index.html')

// 配置引數
const defaultConfig = {
    id: null,               // 視窗唯一id
    background: '#fff',     // 背景色
    route: '',              // 路由地址url
    title: '',              // 標題
    data: null,             // 傳入資料引數
    width: '',              // 視窗寬度
    height: '',             // 視窗高度
    minWidth: '',           // 視窗最小寬度
    minHeight: '',          // 視窗最小高度
    x: '',                  // 視窗相對于螢屏左側坐標
    y: '',                  // 視窗相對于螢屏頂端坐標
    resize: true,           // 是否支持縮放
    maximize: false,        // 最大化視窗
    isMultiWin: false,      // 是否支持多開視窗
    isMainWin: false,       // 是否主視窗
    parent: '',             // 父視窗(需傳入父視窗id)
    modal: false,           // 模態視窗(模態視窗是浮于父視窗上,禁用父視窗)
    alwaysOnTop: false      // 置頂視窗
}

class MultiWindows {
    constructor() {
        // 主視窗
        this.mainWin = null
        // 視窗組
        this.winLs = {}

        // ...
    }

    winOpts() {
        return {
            // 視窗圖示
            icon: join(process.env.ROOT, 'resource/shortcut.ico'),
            backgroundColor: '#fff',
            autoHideMenuBar: true,
            titleBarStyle: 'hidden',
            width: 1000,
            height: 640,
            resizable: true,
            minimizable: true,
            maximizable: true,
            frame: false,
            show: false,
            webPreferences: {
                contextIsolation: true, // 啟用背景關系隔離(為了安全性)(默認true)
                // nodeIntegration: false, // 啟用Node集成(默認false)
                preload: join(process.env.ROOT, 'resource/preload.js'),
                // devTools: true,
                // webSecurity: false
            }
        }
    }

    // 創建新視窗
    createWin(options) {
        const args = Object.assign({}, defaultConfig, options)
        console.log(args)

        // 判斷視窗是否存在
        for(let i in this.winLs) {
            if(this.getWin(i) && this.winLs[i].route === args.route && !this.winLs[i].isMultiWin) {
                this.getWin(i).focus()
                return
            }
        }

        let opt = this.winOpts()
        if(args.parent) {
            opt.parent = this.getWin(args.parent)
        }

        if(typeof args.modal === 'boolean') opt.modal = args.modal
        if(typeof args.resize === 'boolean') opt.resizable = args.resize
        if(typeof args.alwaysOnTop === 'boolean') opt.alwaysOnTop = args.alwaysOnTop
        if(args.background) opt.backgroundColor = args.background
        if(args.width) opt.width = args.width
        if(args.height) opt.height = args.height
        if(args.minWidth) opt.minWidth = args.minWidth
        if(args.minHeight) opt.minHeight = args.minHeight
        if(args.x) opt.x = args.x
        if(args.y) opt.y = args.y

        console.log(opt)

        // 創建視窗物件
        let win = new BrowserWindow(opt)
        // 是否最大化
        if(args.maximize && args.resize) {
            win.maximize()
        }
        this.winLs[win.id] = {
            route: args.route, isMultiWin: args.isMultiWin
        }
        args.id = win.id


        // 加載頁面
        let $url
        if(!args.route) {
            if(process.env.VITE_DEV_SERVER_URL) {
                // 打開開發者除錯工具
                // win.webContents.openDevTools()
    
                $url = process.env.VITE_DEV_SERVER_URL
            }else {
                $url = winURL
            }
        }else {
            $url = `${winURL}#${args.route}`
        }
        win.loadURL($url)
        /*if(process.env.VITE_DEV_SERVER_URL) {
            win.loadURL($url)
        }else {
            win.loadFile($url)
        }*/
        win.webContents.openDevTools()

        win.once('ready-to-show', () => {
            win.show()
        })

        win.on('close', () => win.setOpacity(0))

        // 初始化渲染行程
        win.webContents.on('did-finish-load', () => {
            // win.webContents.send('win-loaded', '加載完成~!')
            win.webContents.send('win-loaded', args)
        })
    }

    // 獲取視窗
    getWin(id) {
        return BrowserWindow.fromId(Number(id))
    }

    // 獲取全部視窗
    getAllWin() {
        return BrowserWindow.getAllWindows()
    }

    // 關閉全部視窗
    closeAllWin() {
        try {
            for(let i in this.winLs) {
                if(this.getWin(i)) {
                    this.getWin(i).close()
                }else {
                    app.quit()
                }
            }
        } catch (error) {
            console.log(error)
        }
    }

    // 開啟主行程監聽
    ipcMainListen() {
        // 設定標題
        ipcMain.on('set-title', (e, data) => {
            const webContents = e.sender
            const wins = BrowserWindow.fromWebContents(webContents)
            wins.setTitle(data)

            // const wins = BrowserWindow.getFocusedWindow()
            // wins.setTitle('啦啦啦')
        })
        // 是否最大化(方法一)
        /*ipcMain.on('isMaximized', e => {
            const win = BrowserWindow.getFocusedWindow()
            e.sender.send('mainReplay', win.isMaximized())
        })*/
        // 是否最大化(方法二)
        ipcMain.handle('isMaximized', (e) => {
            const win = BrowserWindow.getFocusedWindow()
            return win.isMaximized()
        })

        ipcMain.on('min', e => {
            const win = BrowserWindow.getFocusedWindow()
            win.minimize()
        })
        ipcMain.handle('max2min', e => {
            const win = BrowserWindow.getFocusedWindow()
            if(win.isMaximized()) {
                win.unmaximize()
                return false
            }else {
                win.maximize()
                return true
            }
        })
        ipcMain.on('close', (e, data) => {
            // const wins = BrowserWindow.getFocusedWindow()
            // wins.close()
            this.closeAllWin()
        })

        // ...
    }
}

module.exports = MultiWindows

在主行程入口background.js檔案引入封裝視窗,

const { app, BrowserWindow, ipcMain } = require('electron')
const { join } = require('path')

const MultiWindows = require('./src/windows')

// 屏蔽安全警告
// ectron Security Warning (Insecure Content-Security-Policy)
process.env['ELECTRON_DISABLE_SECURITY_WARNINGS'] = 'true'

const createWindow = () => {
    let window = new MultiWindows()

    window.createWin({isMainWin: true})
    window.ipcMainListen()
}

app.whenReady().then(() => {
    createWindow()
    app.on('activate', () => {
        if (BrowserWindow.getAllWindows().length === 0) createWindow()
    })
})

app.on('window-all-closed', () => {
    if (process.platform !== 'darwin') app.quit()
})

在主行程中做一個ipcMain監聽,用來創建獨立視窗,

ipcMain.on('win-create', (event, args) => this.createWin(args))

新建windows/action.js檔案,處理渲染器行程到主行程的異步通信,可以發送同步或異步的訊息到主行程,也可以接收主行程發送的訊息,

/**
 * 創建新視窗
 * @param {object} args | {width: 640, height: 480, route: '/home'}
 */
export function createWin(args) {
    window.electronAPI.send('win-create', args)
}

/**
 * 設定視窗
 * @param {string} type | 'show'/'hide'/'close'/'min'/'max'/'max2min'/'restore'/'reload'
 * @param {number} id
 */
export function setWin(type, id) {
    window.electronAPI.send('win-' + type, id)
}

/**
 * 創建登錄視窗
 */
export function loginWin() {
    createWin({
        isMainWin: true,
        title: '登錄',
        route: '/login',
        width: 550,
        height: 320,
        resize: false,
        alwaysOnTop: true,
    })
}

在vue頁面中呼叫上面封裝的方法,

<template>
    <div class="home">
        ...

        <Button type="success" @click="openWin">打開Manage視窗(設定parent)</Button>
        <Button type="success" @click="openWin1">打開Me視窗(設定resizable/isMultiWin)</Button>
        <Button type="success" @click="openWin2">打開User視窗</Button>
    </div>
</template>

<script>
import { winCfg, createWin } from '@/windows/action'

export default {
    name: 'Home',
    setup() {
        const openWin = () => {
            MessageBox.confirm('提示', '確定打開Manage頁面嗎? 【設定parent屬性】', {
                callback: action => {
                    if(action == 'confirm') {
                        createWin({
                            title: 'Manage.vue',
                            route: '/manage',
                            width: 600,
                            height: 400,
                            background: '#09f',
                            parent: winCfg.window.id,
                            // modal: true
                        })
                    }else if(action == 'cancel') {
                        Message.info('您已取消!')
                    }
                }
            })
        }

        const openWin1 = () => {
            // 左上角
            // let posX = 0
            // let posY = 0

            // 右下角
            let posX = window.screen.availWidth - 850
            let posY = window.screen.availHeight - 600
            MessageBox.confirm('提示', '確定打開Me頁面嗎?', {
                callback: action => {
                    if(action == 'confirm') {
                        createWin({
                            title: 'Me.vue',
                            route: '/me?name=Andy',
                            width: 850,
                            height: 600,
                            x: posX,
                            y: posY,
                            background: 'yellow',
                            resize: false,
                            isMultiWin: true,
                            maximize: true
                        })
                    }else if(action == 'cancel') {
                        Message.info('您已取消!')
                    }
                }
            })
        }

        const openWin2 = () => {
            MessageBox.confirm('提示', '確定打開User頁面嗎?', {
                callback: action => {
                    if(action == 'confirm') {
                        createWin({
                            title: 'User.vue',
                            route: '/user',
                            width: 700,
                            height: 550,
                            minWidth: 300,
                            minHeight: 300,
                            data: {
                                name: 'Andy',
                                age: 20
                            },
                            background: 'green',
                            isMultiWin: true
                        })
                    }else if(action == 'cancel') {
                        Message.info('您已取消!')
                    }
                }
            })
        }

        // ...

        return {
            openWin,
            openWin1,
            openWin2,

            // ...
        }
    }
}
</script>

設定 frame: false 創建無邊框視窗,

設定 -webkit-app-region: drag 來實作自定義拖拽區域,設定后的按鈕操作無法回應其它事件,只需設定 -webkit-app-region: no-drag 即可實作回應事件,

electron+vite提供的一些環境變數,

process.env.NODE_ENV
process.env.VITE_DEV_SERVER_URL

在開發環境,加載vite url,生產環境,則加載vite build出來的html,

Ok,綜上就是electron25+vite4結合構建跨端應用的一些分享,希望對大家有所幫助哈~~

 

本文為博主原創文章,未經博主允許不得轉載,歡迎大家一起交流 QQ(282310962) wx(xy190310)

轉載請註明出處,本文鏈接:https://www.uj5u.com/qiye/553883.html

標籤:其他

上一篇:JS中的事件監聽

下一篇:返回列表

標籤雲
其他(160043) Python(38189) JavaScript(25466) Java(18161) C(15234) 區塊鏈(8268) C#(7972) AI(7469) 爪哇(7425) MySQL(7217) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5873) 数组(5741) R(5409) Linux(5344) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4579) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2434) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1977) 功能(1967) Web開發(1951) HtmlCss(1950) C++(1927) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1878) .NETCore(1862) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • IEEE1588PTP在數字化變電站時鐘同步方面的應用

    IEEE1588ptp在數字化變電站時鐘同步方面的應用 京準電子科技官微——ahjzsz 一、電力系統時間同步基本概況 隨著對IEC 61850標準研究的不斷深入,國內外學者提出基于IEC61850通信標準體系建設數字化變電站的發展思路。數字化變電站與常規變電站的顯著區別在于程序層傳統的電流/電壓互 ......

    uj5u.com 2020-09-10 03:51:52 more
  • HTTP request smuggling CL.TE

    CL.TE 簡介 前端通過Content-Length處理請求,通過反向代理或者負載均衡將請求轉發到后端,后端Transfer-Encoding優先級較高,以TE處理請求造成安全問題。 檢測 發送如下資料包 POST / HTTP/1.1 Host: ac391f7e1e9af821806e890 ......

    uj5u.com 2020-09-10 03:52:11 more
  • 網路滲透資料大全單——漏洞庫篇

    網路滲透資料大全單——漏洞庫篇漏洞庫 NVD ——美國國家漏洞庫 →http://nvd.nist.gov/。 CERT ——美國國家應急回應中心 →https://www.us-cert.gov/ OSVDB ——開源漏洞庫 →http://osvdb.org Bugtraq ——賽門鐵克 →ht ......

    uj5u.com 2020-09-10 03:52:15 more
  • 京準講述NTP時鐘服務器應用及原理

    京準講述NTP時鐘服務器應用及原理京準講述NTP時鐘服務器應用及原理 安徽京準電子科技官微——ahjzsz 北斗授時原理 授時是指接識訓通過某種方式獲得本地時間與北斗標準時間的鐘差,然后調整本地時鐘使時差控制在一定的精度范圍內。 衛星導航系統通常由三部分組成:導航授時衛星、地面檢測校正維護系統和用戶 ......

    uj5u.com 2020-09-10 03:52:25 more
  • 利用北斗衛星系統設計NTP網路時間服務器

    利用北斗衛星系統設計NTP網路時間服務器 利用北斗衛星系統設計NTP網路時間服務器 安徽京準電子科技官微——ahjzsz 概述 NTP網路時間服務器是一款支持NTP和SNTP網路時間同步協議,高精度、大容量、高品質的高科技時鐘產品。 NTP網路時間服務器設備采用冗余架構設計,高精度時鐘直接來源于北斗 ......

    uj5u.com 2020-09-10 03:52:35 more
  • 詳細解讀電力系統各種對時方式

    詳細解讀電力系統各種對時方式 詳細解讀電力系統各種對時方式 安徽京準電子科技官微——ahjzsz,更多資料請添加VX 衛星同步時鐘是我京準公司開發研制的應用衛星授時時技術的標準時間顯示和發送的裝置,該裝置以M國全球定位系統(GLOBAL POSITIONING SYSTEM,縮寫為GPS)或者我國北 ......

    uj5u.com 2020-09-10 03:52:45 more
  • 如何保證外包團隊接入企業內網安全

    不管企業規模的大小,只要企業想省錢,那么企業的某些服務就一定會采用外包的形式,然而看似美好又經濟的策略,其實也有不好的一面。下面我通過安全的角度來聊聊使用外包團的安全隱患問題。 先看看什么服務會使用外包的,最常見的就是話務/客服這種需要大量重復性、無技術性的服務,或者是一些銷售外包、特殊的職能外包等 ......

    uj5u.com 2020-09-10 03:52:57 more
  • PHP漏洞之【整型數字型SQL注入】

    0x01 什么是SQL注入 SQL是一種注入攻擊,通過前端帶入后端資料庫進行惡意的SQL陳述句查詢。 0x02 SQL整型注入原理 SQL注入一般發生在動態網站URL地址里,當然也會發生在其它地發,如登錄框等等也會存在注入,只要是和資料庫打交道的地方都有可能存在。 如這里http://192.168. ......

    uj5u.com 2020-09-10 03:55:40 more
  • [GXYCTF2019]禁止套娃

    git泄露獲取原始碼 使用GET傳參,引數為exp 經過三層過濾執行 第一層過濾偽協議,第二層過濾帶引數的函式,第三層過濾一些函式 preg_replace('/[a-z,_]+\((?R)?\)/', NULL, $_GET['exp'] (?R)參考當前正則運算式,相當于匹配函式里的引數 因此傳遞 ......

    uj5u.com 2020-09-10 03:56:07 more
  • 等保2.0實施流程

    流程 結論 ......

    uj5u.com 2020-09-10 03:56:16 more
最新发布
  • 基于electron25+vite4創建多視窗|vue3+electron25新開模態表單

    在寫這篇文章的時候,查看了下electron最新穩定版本由幾天前24.4.0升級到了25了,不得不說electron團隊迭代速度之快! 前幾天有分享一篇electron24整合vite4全家桶技術構建桌面端vue3應用示例程式。 https://www.cnblogs.com/xiaoyan2017 ......

    uj5u.com 2023-05-31 08:42:23 more
  • JS中的事件監聽

    JavaScript的事件監聽是一種機制,用于在HTML檔案中的元素上注冊事件處理程式,以便在特定事件發生時執行相應的JavaScript代碼。 事件監聽的基本思想是將事件處理程式(也稱為事件回呼函式)系結到特定的事件上。所謂特定的事件是當用戶與頁面進行互動時,比如點擊按鈕、鍵盤輸入、滑鼠移動等,瀏 ......

    uj5u.com 2023-05-31 08:42:10 more
  • Java實作打包壓縮檔案或檔案夾生成zip以實作多檔案批量下載

    有時候在系統中需要一次性下載多個檔案,但逐個下載檔案比較麻煩。這時候,最好的解決辦法是將所有檔案打包成一個壓縮檔案,然后下載這個壓縮檔案,這樣就可以一次性獲取所有所需的檔案了。 下面是一個名為CompressUtil的工具類的代碼,它提供了一些方法來處理檔案壓縮和下載操作: ```java impo ......

    uj5u.com 2023-05-31 08:41:58 more
  • JS中的事件監聽

    JavaScript的事件監聽是一種機制,用于在HTML檔案中的元素上注冊事件處理程式,以便在特定事件發生時執行相應的JavaScript代碼。 事件監聽的基本思想是將事件處理程式(也稱為事件回呼函式)系結到特定的事件上。所謂特定的事件是當用戶與頁面進行互動時,比如點擊按鈕、鍵盤輸入、滑鼠移動等,瀏 ......

    uj5u.com 2023-05-31 08:41:53 more
  • ENVI指定像元數量(行數與列數)裁剪柵格影像

    本文介紹基于**ENVI**軟體,實作柵格遙感影像按照**像元行列號與個數**進行**指定矩形區域裁剪**的方法。 一般的,如果我們需要裁剪某個具體的行政區域,按照對應區域的矢量圖層裁剪即可;如果需要裁剪某個大致的區域范圍,可以按照文章[ArcMap手動新建矢量要素的方式](https://www. ......

    uj5u.com 2023-05-31 08:40:49 more
  • ENVI指定像元數量(行數與列數)裁剪柵格影像

    本文介紹基于**ENVI**軟體,實作柵格遙感影像按照**像元行列號與個數**進行**指定矩形區域裁剪**的方法。 一般的,如果我們需要裁剪某個具體的行政區域,按照對應區域的矢量圖層裁剪即可;如果需要裁剪某個大致的區域范圍,可以按照文章[ArcMap手動新建矢量要素的方式](https://www. ......

    uj5u.com 2023-05-31 08:39:46 more
  • 【一步步開發AI運動小程式】九、姿態輔助除錯桌面工具折使用

    > 隨著人工智能技術的不斷發展,阿里體育等IT大廠,推出的“樂動力”、“天天跳繩”AI運動APP,讓**云上運動會、線上運動會、健身打卡、AI體育指導**等概念空前火熱。那么,能否將這些在APP成功應用的場景搬上小程式,分享這些概念的紅利呢?本系列文章就帶您一步一步從零開始開發一個AI運動小程式,本 ......

    uj5u.com 2023-05-30 07:56:48 more
  • JavaScript全決議——Express框架介紹與入門

    本文為千鋒資深前端教學老師帶來的【JavaScript全決議】系列,文章內含豐富的代碼案例及配圖,從0到1講解JavaScript相關知識點,致力于教會每一個人學會JS!文末有本文重點總結,可以收藏慢慢看\~ 更多技術類內容,主頁關注一波! ......

    uj5u.com 2023-05-30 07:56:43 more
  • html+css實作二級導航欄效果,簡單易看懂噢!

    這應該是這幾天以來看到的最簡單易懂的有二級選單欄的導航欄效果實作了 使用html+css實作,看了好幾天導航欄的實作方式,要么是太復雜的需要JS或者框架的,要么是太簡單僅僅使用div和超鏈接的, 再要么就是只有簡單的一級導航,沒有二級選單欄的,心情復雜 就想找一個有二級選單欄,使用html+css實 ......

    uj5u.com 2023-05-30 07:56:37 more
  • 【一步步開發AI運動小程式】九、姿態輔助除錯桌面工具折使用

    > 隨著人工智能技術的不斷發展,阿里體育等IT大廠,推出的“樂動力”、“天天跳繩”AI運動APP,讓**云上運動會、線上運動會、健身打卡、AI體育指導**等概念空前火熱。那么,能否將這些在APP成功應用的場景搬上小程式,分享這些概念的紅利呢?本系列文章就帶您一步一步從零開始開發一個AI運動小程式,本 ......

    uj5u.com 2023-05-30 07:56:19 more