Electron 不錯,但也不是完美的。 Electron 帶來了很多優秀的桌面軟體,但並不一定總是適合我們的需求。 多個選擇總是好事! ▶ 我使用 Electron 遇到的一些麻煩 1、Electron 太大了! 2、每一個 Electron 寫的軟體都要重覆地帶一個 Electron …… 升級 ...
Electron 不錯,但也不是完美的。
Electron 帶來了很多優秀的桌面軟體,但並不一定總是適合我們的需求。
多個選擇總是好事!
▶ 我使用 Electron 遇到的一些麻煩
1、Electron 太大了!
2、每一個 Electron 寫的軟體都要重覆地帶一個 Electron …… 升級與分發都不方便。
3、Electron 不方便嵌入其他視窗界面,與其他語言、技術融合不易。
4、並不是所有桌面軟體都需要 Electron 的跨平臺特性。macOS , Linux 的桌面系統市場份額小於被遺忘的 Windows 8 ,如果軟體只是在 Windows 平臺運行,並且需要大量與專用系統 API 交互,跨平臺反而是不必要的負擔。
5、我曾經在 aardio 中封裝了一個 electron 擴展庫,然後我在寫這個擴展庫的時候,當時看到的還是 remote 真香 …… 然後我為這個擴展庫寫了個很大的 JS 文件就用到了 remote。可是等我寫完沒多久, 就看到 remote 被 Electron 拋棄了,remote 會慢一萬倍 ,各種缺陷 ……
▶ WebView2 的優勢
1、WebView2 基於性能強悍的 Edge(Chromium) 內核。
2、調用 WebView2 生成的軟體體積很小。所有基於 WebView2 的軟體可以共用同一個 WebView2 組件。Win11 已經內置 WebView2 組件,其他操作系統也可以快速地自動安裝 WebView2 。
3、WebView2 介面非常簡潔,嵌入其他視窗界面也非常方便。
總結一句話就是:WebView2 簡單、好用、生成軟體體積小。
aardio 標準庫中的 web.view 就是基於 WebView2。WebView2 的介面是如此簡潔,所以我寫的這個庫也只有很少的代碼。aardio 支持在低版本操作系統自動檢測與安裝 WebView2 ( 速度極快,不用加任何代碼。隨著 WebView2 普及,舊系統減少,需要安裝的系統也會越來越少 ),又因為 aardio 可以將網頁自動內嵌到獨立 EXE 文件,就可以非常方便地生成獨立 EXE 程式。
▶ 一個最簡單的程式演示
下麵我們用 aardio 調用 web.view (WebView2)寫一個最簡單的程式:
import win.ui; /*DSG{{*/ mainForm = win.form(text="WebView2") mainForm.add( btnCallJs={cls="button";text="調用 JS 函數";left=461;top=395;right=726;bottom=449;note="點這裡調用 JavaScript 函數";z=1}; custom={cls="custom";left=17;top=21;right=730;bottom=356;z=2} ) /*}}*/ //創建瀏覽器組件 import web.view; var wb = web.view(mainForm.custom); //導出本地函數給網頁 JavaScript wb.external = { getComputerName = function(){ return sys.getComputerName(); } } import sys; //寫入網頁 HTML wb.html = /** <html> <head> <script> (async ()=>{ var n = await aardio.getComputerName(); alert(n); })() </script> </head> <body> **/ //響應按鈕事件 mainForm.btnCallJs.oncommand = function(id,event){ //調用 JS 函數 wb.xcall("document.write","測試") } mainForm.show(); win.loopMessage();
對,這就是一個完整程式的源代碼,可以一鍵生成獨立 EXE 文件。
▶ 入門
首先點選 「aardio 主菜單 > 新建工程 > 視窗程式 > 空白工程」,然後點擊「創建工程」。
如果熟悉網頁前端開發,也可以點擊 「 新建工程 > Web 界面 > WebView2 」創建工程。
雙擊工程入口代碼 main.aardio 打開主視窗,自「界面控制項」中拖一個 「調用 JS 函數」的按鈕上去,再拖一個 custom 控制項到窗體上 —— 用來嵌入網頁:
然後切換到代碼視圖,添加以下代碼創建網頁瀏覽器:
import web.view; var wb = web.view(mainForm.custom);
web.view 的第 1 個參數指定要嵌入 WebView2 的視窗對象,該參數可以是 mainForm.custom 這樣的控制項視窗,也可以是 mainForm 這樣的窗體對象。
下麵使用
wb.html = "<html></html>"
就可以寫網頁 HTML 代碼了。
或者使用
wb.go("網址")
可以打開指定的網頁。
使用
import wsock.tcp.simpleHttpServer;
wb.go("\res\index.html");
可以打開資源目錄的網頁,支持SPA 單頁應用。資源目錄可以嵌入 EXE 生成 獨立 EXE 文件,放心不用多寫其他代碼。
添加下麵的代碼導出 external 對象給網頁 JavaScript :
//導出本地函數給網頁 JavaScript wb.external = { getComputerName = function(){ return sys.getComputerName(); } } import sys;
在網頁 JavaScript 里可以調用上面導出的 external 對象,不過在 JavaScript 里要用 aardio 這個名字表示 external 對象,網頁代碼如下:
wb.html = /** <html> <head> <script> (async ()=>{ var n = await aardio.getComputerName(); alert(n); })() </script> </head> <body> **/
註意在 aardio 中 /* 註釋 */ 可以作為字元串賦值給其他變數,請參考:aardio 編程語言快速入門——語法速覽
要註意所有 aardio 對象在 JavaScript 中都是非同步 Promise 對象。如上在 async 函數體內可以愉快地使用 await 調用 aardio 函數 —— 這非常方便。
我們在窗體設計視圖雙擊「調用 JS 函數」按鈕,這會切換到代碼視圖,並自動添加以下回調函數:
mainForm.btnCallJs.oncommand = function(id,event){ }
用戶點擊按鈕時就會調用上面的函數。
小改一下添加 aardio 代碼調用 JavaScript 函數:
//響應按鈕事件 mainForm.btnCallJs.oncommand = function(id,event){ //調用 JS 函數 wb.xcall("document.write","測試") }
很簡單,一個程式就寫好了。可以在 aardio 中點擊「運行」按鈕直接運行代碼,也可以點擊「發佈」按鈕直接生成 EXE 文件。
▶ 如何將網頁顯示在窗體的指定位置?並且支持自動縮放?
web.view() 構造函數的第 1 個嵌入視窗參數可以是 win.form 對象(獨立視窗),也可以是 custom, static 這樣的普通控制項對象。例如前面的例子就是將 WebView2 嵌入 custom 控制項:
import web.view; var wb = web.view(mainForm.custom);
aardio 中的所有控制項都可以非常方便的支持自動縮放。只要簡單地在窗體設計器中選定控制項,然後在「屬性」面板設置「固定邊距」、「自適應大小」這些屬性就可以。
▶ 使用 wb.export 導出 aardio 函數到 Javascript
前面我們介紹過使用 external 導出 aardio 函數到網頁 JavaScript 。我們還可以用 wb.export 導出 aardio 函數,先看例子:
import web.view; var wb = web.view(mainForm.custom); wb.export({ alert = function(msg){ winform.msgbox(msg) }; nativeAdd = function(a,b){ return a + b; } })
註意:
1、wb.export() 導出的是 JavaScript 全局函數。
2、wb.export() 導出的函數在 JavaScript 中同樣是非同步 Promise 對象。
3、wb.export() 導出的 Javascript 全局函數, 使用 JSON 自動轉換調用參數和返回值,可以更好的相容只能支持純 aardio 對象 / 純 JavaScript 對象的代碼。
4、wb.export() 導出的函數內部禁止調用 wb.doScript 或 wb.eval 執行Javascript 。
wb.external 內部是調用 wb.exportHostObject() 導出 aardio 對象,中間不需要經過 JSON 自動轉換。
▶ 示例:網頁 JavaScript 調用本地 Ping 命令
我經常被問到幾個類似的問題:
1、JavaScript 的非同步函數太麻煩了,怎樣把他搞成同步的,不用 await ,不用 async 。
2、JavaScript 的非同步函數太好用了,怎樣在 aardio 中也這樣搞,如何在 aardio 里 await 。
其實同步有同步的優勢,非同步有非同步的好處,揚長避短是智慧,倒行逆施最累人。下麵我們一起來寫一個在 WebView2 中調用本地 Ping 命令的小程式體驗一下。
第一步:創建視窗。
import win.ui; var winform = win.form(text="Ping")
第二步:基於視窗創建 WebView2 瀏覽器組件。
import web.view; var wb = web.view(winform);
第三步:使用 external 對象導出 JavaScript 可以調用的本地函數。
import process.popen; wb.external = { ping = function(domain){ var prcs = process.popen("ping "+ domain); for( all,out,err in prcs.each() ){ wb.invoke("document.body.insertAdjacentText",'beforeend',all); } return "恭喜,事做好了!" } }
在 JavaScript 里用 aardio.ping() 就可以直接調用上面的 external.ping() 函數了。
第四步:下麵在網頁里寫 JavaScript 來調用 aardio 函數。
wb.html = /** <body style="white-space: pre;"><script> doSomething = async() => { var result = await aardio.ping('www.baidu.com'); document.body.insertAdjacentText('beforeend',result); }; </script> <button onclick="doSomething()">開始幹活了</ping> **/
就這麼短短幾句,一個簡單的程式就完成了,請看運行效果:
上面程式的完整 aardio 源代碼如下:
//創建視窗 import win.ui; var winform = win.form(text="Ping") //嵌入瀏覽器組件 import web.view; var wb = web.view(winform); //導出 aardio 函數到 JavaScript wb.external = { ping = function(domain){ //同步有同步的優勢,揚長避短是智慧,倒行逆施最累人。 var prcs = process.popen("ping "+ domain); for( all,out,err in prcs.each() ){ wb.invoke("document.body.insertAdjacentText",'beforeend',all); } return "恭喜,事做好了!" } } import process.popen; //寫入網頁 HTML wb.html = /** <body style="white-space: pre;"><script> doSomething = async() => { //非同步有非同步的好處,揚長避短是智慧,倒行逆施最累人。 var result = await aardio.ping('www.baidu.com'); document.body.insertAdjacentText('beforeend',result); }; </script> <button onclick="doSomething()">開始幹活了</ping> **/ //顯示視窗 winform.show(); //啟動界面消息迴圈 win.loopMessage();
▶ aardio 調用 JS 函數
在 aardio 中可以使用 wb.doScript() , wb.eval() , wb.xcall() 等函數調用網頁 JavaScript ,下麵看一個在 aardio 中調用 xterm.js 的簡單例子:
import win.ui; var winform = win.form(text="xterm") import web.view; var wb = web.view(winform); wb.html = /** <!DOCTYPE html> <head> <meta charset="UTF-8"> <title></title> <link rel="stylesheet" href="https://unpkg.com/[email protected]/css/xterm.css"> <script src="https://unpkg.com/[email protected]/lib/xterm.js"></script> </head> <body style="height:100vh;"> <script> let term = new Terminal(); term.open(document.body); term.write('\x1b[31m紅色字體\x1b[37m測試') </script> </body> </html> **/ wb.xcall("term.write",'\e[32m綠色字體'); winform.show(); win.loopMessage();
▶ 無邊框視窗:用網頁實現視窗標題欄
「無邊框視窗」指的是去掉獨立窗體預設的邊框與標題欄,然後由程式自行定製邊框與標題欄。
aardio 做這事還是很容易的,首頁在窗體屬性中指定「邊框」屬性為 none。
這樣直接運行後顯示的窗體就沒有邊框和標題欄了( 按 Alt + F4 關閉視窗 )。
然後添加下麵的代碼就可以為窗體添加標題欄、標題欄按鈕、陰影邊框、並支持拖動邊框縮放:
import win.ui.simpleWindow;
win.ui.simpleWindow(winform);
win.ui.simpleWindow 的源碼很簡單,參考其源碼也可以自己編寫新的庫定製邊框與標題欄。
這裡我們不用上面的方法,而是用網頁實現標題欄。
我們知道網頁繪製一個標題欄與標題欄按鈕很簡單,難點在於怎麼在網頁里控制視窗。我們先學習幾個專用於無邊框視窗的 aardio 函數:
winform.hitMax() //模擬點擊最大化按鈕 winform.hitMin() //模擬點擊最小化按鈕 winform.hitClose() //模擬點擊關閉按鈕 winform.hitCaption() //拖動標題欄
下麵寫個簡單的例子,先看下運行效果:
WebView2 無邊框視窗示例完整源碼如下:
import win.ui; /*DSG{{*/ var winform = win.form(text="無邊框視窗";right=759;bottom=469;bgcolor=16777215;border="none") winform.add() /*}}*/ import web.view; var wb = web.view(winform); //導出為 Javascript 中的 aardio 對象 wb.external = { close = function(){ winform.close(); }; hitCaption = function(){ winform.hitCaption(); }; hitMin = function(){ winform.hitMin(); }; hitMax = function(){ return winform.hitMax(); }; } wb.html = /** <!doctype html> <html> <head> <meta charset="utf-8"> <style type="text/css"> html { margin: 0px; padding: 0px; background-color: #202020; } #title-bar { height: 32px; padding: 0px; margin: 0px; } #title-bar .caption { position: fixed; top: 0px; left: 0px; width: 100%; padding-left: 10px; color: #ADADAD; line-height: 32px; font-size: 14px; cursor: default; user-select:none; } #title-bar .buttons { position: fixed; top: 1px; right: 1px; } #title-bar button { font: 14px Marlett ; color: #F5F5F5; background-color: transparent; border: none; height: 28px; width: 28px; } #title-bar button:hover { background-color: #FF4500; } #title-bar button:active { background-color: #B0451E; color: #C5C5C5; } #main { padding: 12px; color: #C0C0C0; } </style> <script type="text/javascript"> </script> </head> <body> <div id="title-bar" > <div class="caption" onmousedown="aardio.hitCaption()">按住這裡調用 aardio.hitCaption() 拖動視窗 </div> <div class="buttons"> <button id="min-btn" onclick="aardio.hitMin()">0</button> <button id="max-btn" onclick="aardio.hitMax()">1</button> <button id="close-btn" onclick="aardio.close()">r</button> </div> </div> <div id="main"> 1、請指定窗體「邊框」屬性為 none ,創建無邊框視窗。<br /> 2、調用 win.ui.shadow(winform) 創建陰影邊框<br /> </div> <script src="default.js"></script> </body> </html> **/ //添加陰影邊框 import win.ui.shadow; win.ui.shadow(winform); //設置視窗縮放範圍 import win.ui.minmax; win.ui.minmax(winform); //切換最大化、還原按鈕 winform.adjust = function( cx,cy,wParam ) { if( wParam == 0x2/*_SIZE_MAXIMIZED*/ ){ wb.doScript(`document.getElementById("max-btn").innerText="2";`) } elseif( wParam == 0x0/*_SIZE_RESTORED*/ ){ wb.doScript(`document.getElementById("max-btn").innerText="1";`) } }; winform.show(); win.loopMessage();
以上源碼來自 aardio 自帶範例 > Web 界面 > web.view :
▶ WebView2 + 前端工程
如果熟悉網頁前端開發,也可以點擊 「 新建工程 > Web 界面 > WebView2 」創建工程。
運行創建的範例工程會顯示幫助:
這些熟悉前端的一看就懂,就不多說了。
註意 WebView2 預設工程的「網頁源碼」這個目錄的「內嵌資源」屬性為 false —— 也就是說發佈後的 EXE 文件不會包含這個目錄。
而工程中的「網頁」目錄「內嵌資源」屬性為 true —— 也就是說發佈後的 EXE 文件會包含這個目錄。
「網頁」目錄「本地構建」屬性為 true —— 這指的是該目錄下的文件會無條件添加到發佈 EXE 文件中(不必添加到工程 )。
▶ 其他瀏覽器組件
aardio 中的瀏覽器組件非常多,用法與 web.view 基本都類似。aardio 甚至可以調用操作系統已安裝的 Chrome,Edge 等瀏覽器寫軟體界面。
請參考「 aardio 自帶範例 > Web 界面」: