摘要:使用實(shí)現(xiàn)桌面應(yīng)用實(shí)現(xiàn)離線可用很多方法,比如使用技術(shù)。還有一個(gè)好處,因?yàn)樗耆趤?lái)實(shí)現(xiàn)可以使用的一些新功能,那我們理論上可以在做桌面應(yīng)用時(shí)順手把應(yīng)用也做了。
本文將會(huì)講述一個(gè)完整的跨端桌面應(yīng)用?代碼畫板?的構(gòu)建,會(huì)涉及到整個(gè)軟件開發(fā)流程,從開始的設(shè)計(jì)、編碼、到最后產(chǎn)品成型、包裝等。
本文不僅僅是一篇技術(shù)方面的專業(yè)文章,更會(huì)有很多產(chǎn)品方面的設(shè)計(jì)思想和將技術(shù)轉(zhuǎn)換成生產(chǎn)力的思考,我將結(jié)合我自己的使用場(chǎng)景完全的講解整個(gè)開發(fā)流程,當(dāng)然涉及到設(shè)計(jì)方面的不一定具有普遍實(shí)用性,多數(shù)情況下都是我自己的一些喜好,我只關(guān)心自己的需求。
同時(shí)本文只從整體上講思路,也會(huì)有個(gè)別的技術(shù)細(xì)節(jié)和常規(guī)套路,有興趣的也可以直接去 github 上看 源碼,文章會(huì)比較長(zhǎng),如果你只想知道一些拿來(lái)即用的「干貨」,或許這篇文章并不是一個(gè)好的選擇
一、定位需求事情的起因是這樣的,因?yàn)槲覀儍?nèi)部會(huì)有一些培訓(xùn)會(huì)議。會(huì)經(jīng)?,F(xiàn)場(chǎng)演示一些代碼片段。比如說(shuō)我們講到 React 的時(shí)候會(huì)現(xiàn)場(chǎng)寫一些組件,讓大家能直觀的感受到 React 的一些功能。
但是通常由于條件所所限,會(huì)議總會(huì)遇到一些意外。比如斷網(wǎng)、投影分辨率低看不清文字等
起初我們用的是在線版的 codepen,但是感覺并不是那么好用。比如不能方便的修改字體大小,必須要在連網(wǎng)的情況下才能使用。另外它的 UI 設(shè)計(jì)不是很緊湊,通常我們展示代碼的時(shí)候都投影是寸土寸金的,應(yīng)該有一個(gè)簡(jiǎn)潔又不失功能的 UI 界面,能全屏展示…
于是我解決自己實(shí)現(xiàn)一個(gè)這樣的輪子,那么大概的需求目標(biāo)是有了:
離線可用
可以改變界面字體大小
更加簡(jiǎn)潔的 UI
…
二、整體設(shè)計(jì) 應(yīng)用風(fēng)格代碼畫板解決的是?臨時(shí)性 的一些?演示代碼?的需求,所以它的本質(zhì)屬性是一個(gè)拿來(lái)即用的工具,它不應(yīng)該有更復(fù)雜的功能,比如用戶登錄、代碼片段的管理等。這些需求不是它要解決的。代碼畫板會(huì)提供一個(gè)簡(jiǎn)單的導(dǎo)出成 HTML 文件的功能,可以方便用戶存儲(chǔ)整個(gè) HTML 文件。
既然是用來(lái)演示代碼的,那么它的界面上應(yīng)該只有兩個(gè)東西,一個(gè)是 代碼,一個(gè)就是?預(yù)覽。像代碼/控制臺(tái)切換的功能都做成 tab 的形式,正常情況不需要讓他們展示出來(lái)。像 codepen 那樣把所有的代碼編輯器功能都展示出來(lái)我認(rèn)為是不對(duì)的。
codepen 的界面給人感覺非常復(fù)雜,有很多功能點(diǎn)。當(dāng)然我并不是在批評(píng)它,codepen 做為一個(gè)需要商業(yè)化運(yùn)營(yíng)的軟件,勢(shì)必會(huì)做的非常復(fù)雜,這樣才能滿足更多用戶的需求。然而程序員寫軟件則可以完全按照自己的想法來(lái),哪怕這個(gè)應(yīng)用只給自己一個(gè)人用呢。
桌面應(yīng)用的設(shè)計(jì)桌面應(yīng)用的設(shè)計(jì)和 web 界面的設(shè)計(jì)還是有些細(xì)微區(qū)別的,同樣的基于 electron 的應(yīng)用,有的應(yīng)用會(huì)讓人感覺很「原生」,有的則一眼就能看出來(lái)是用 CSS 畫的。我在設(shè)計(jì)代碼畫板的時(shí)候也盡量向原生靠近,避免產(chǎn)生落差感。比如禁用鼠標(biāo)手型圖標(biāo)、在按鈕或者非可選元素上禁止用戶選擇:
cursor: default; user-select: none
因?yàn)閷?shí)際上用戶在使用一款應(yīng)用的時(shí)候感性的因素影響占很大一部分,比如說(shuō)有人不喜歡 electron 可能就是因?yàn)榭吹竭^(guò) electron 里面嵌一個(gè)完整的 web 頁(yè)面的操作,這就讓人很反感。但是這不是 electron 的問(wèn)題,而是應(yīng)用設(shè)計(jì)者的問(wèn)題。
應(yīng)用標(biāo)識(shí)的設(shè)計(jì)說(shuō)實(shí)話應(yīng)用 logo 設(shè)計(jì)我也是業(yè)余水平,但是聊勝于無(wú)。既然水平不行,那就盡量設(shè)計(jì)的不難看就行了。可以參考一些好的設(shè)計(jì)。我用 sketch 畫出 logo 的外形,sketch 有很多 macOS 的模塊可以從網(wǎng)上下載下來(lái),直接基于模板修改就可以了。
代碼畫板主要的界面是分割開的兩個(gè)面板,左邊是代碼,右邊是預(yù)覽。所以我就大概畫了一個(gè)形狀
這個(gè) logo 有個(gè)問(wèn)題就是線條過(guò)多,小尺寸的時(shí)候看不清楚。這個(gè)問(wèn)題我暫時(shí)先忽略了,畢竟我還不是專業(yè)的,后續(xù)有好的創(chuàng)意可以再改
默認(rèn)設(shè)置代碼畫板也 不會(huì)有 設(shè)置界面,因?yàn)槌S玫脑O(shè)置都預(yù)定義好了,你不需要配置。頂多改變下代碼字體的大小。使用編輯器的通用快捷鍵 command++/-?就解決了,或者插入三方庫(kù),直接使用編輯器的通用命令快捷鍵 command+p?調(diào)出。我們的思路就是把復(fù)雜的東西幫用戶隱藏在后臺(tái),觀眾只需要關(guān)注演員臺(tái)上的一分鐘,而不必了解其它細(xì)節(jié)。
快捷鍵/可用性由于代碼畫板的界面非常簡(jiǎn)單,在一些細(xì)小的必要功能就得添加一些快捷鍵。比如:切換 HTML/CSS/JS/Console 代碼編輯器,我在每個(gè) tab 上加了數(shù)字標(biāo)號(hào),暗示它是有順序有快捷鍵的,而且這個(gè)切換方式和 Chrome tab 切換的邏輯一致,使用 command+數(shù)字?就可以實(shí)現(xiàn),萬(wàn)一還是有人不會(huì)用的話,可以去看幫助文檔。里面有所有的快捷鍵。
界面中間的分割條可以自定義拖動(dòng),雙擊重置平分界面
剛開始的時(shí)候我把每個(gè) tab 頁(yè)簽都分割成多帶帶的面板,因?yàn)槲矣X得這個(gè)能拖動(dòng)自定義面板大小的交互實(shí)在是太爽了,忍不住想去拖動(dòng)它。但是后來(lái)想想,其實(shí)并沒有必要,我們寫代碼時(shí)應(yīng)該更專注于代碼本身,如果只有兩個(gè)面板,那么這個(gè)界面無(wú)論是認(rèn)知還是使用起來(lái)就沒有任何困難。
因?yàn)槲覀儾⒉恍枰岩欢训墓δ艿慕缑嫠そo用戶,讓他們自己去選擇。
三、技術(shù)調(diào)研 實(shí)現(xiàn)控制臺(tái)通過(guò)使用流行的幾款在線代碼運(yùn)行工具,我發(fā)現(xiàn)他們有一個(gè)共同的問(wèn)題:控制臺(tái)很難用。無(wú)法像 Chrome Console 那樣展示任意類型的 JS 值。比如我想 log 一段嵌套的 JS 對(duì)象:
console.log({ a: { b: 1, c: { d: [1, 2, 3] } }})
大多數(shù)都展示成這樣的:
[object Object] { a: [object Object] { b: 1 } }
Chrome 是這樣的:
顯然 Chrome 控制臺(tái)中更直觀。所以我們需要在前面的基礎(chǔ)上加一個(gè)需求,即:實(shí)現(xiàn)一個(gè)基于 DOM 的日志展示界面(無(wú)限級(jí)聯(lián)選擇)
日志界面應(yīng)該有下面這些功能:
展示任意 JS 類型的數(shù)據(jù)
Primitive 類型的數(shù)據(jù)顯示不同的顏色(number - 藍(lán)色,string - 綠色)
Object 類型默認(rèn)折疊起來(lái),點(diǎn)擊按鈕展示子級(jí),屬性過(guò)多需要展示縮略信息
數(shù)組前應(yīng)該有長(zhǎng)度標(biāo)記
能展示 JS 運(yùn)行時(shí)的報(bào)錯(cuò) Error 信息
集成現(xiàn)代化的前端框工作流現(xiàn)代化的前端寫頁(yè)面肯定不是 HTML/CSS/JS 一把梭了,至少應(yīng)該有 Sass/Babel 的支持吧。
Sass 嵌套能讓你少寫很多選擇器,當(dāng)然 Less 也可以,但是在我們的這個(gè)應(yīng)用里面區(qū)別不大,一般來(lái)說(shuō)臨時(shí)性的寫一些代碼很少會(huì)用到它們的細(xì)節(jié)功能。有?變量?和?選擇器?嵌套就夠了
Babel 主要是解決了寫 React 的問(wèn)題,不用再安裝一大堆的構(gòu)建工具了,直接使用 UMD 的 React/ReactDOM 就可以了,而且 electron 內(nèi)嵌的 chromium 也支持了 es6 的 class 寫法,實(shí)際上 Babel 主要的目的還是用來(lái)轉(zhuǎn)譯 JSX
注意這里是有一個(gè)我認(rèn)為是?剛性?的需求,比如臨時(shí)忽然有個(gè)想法,或者想驗(yàn)證一段代碼的話,正常情況是使用你的編輯器,新建 demo.html/demo.css/demo.js 等這些操作。但是這些動(dòng)作太浪費(fèi)時(shí)間了。有了代碼畫板以后,直接打應(yīng)用就可以開始 coding 了,真正能做到開箱即用。
提高程序的擴(kuò)展性我們?cè)趯?demo 頁(yè)面時(shí)通常是要引用很多第三方類庫(kù)的,比如:Bootstrp/jQuery 等。我希望有一種方法可以方便的引用到這些庫(kù),直接把庫(kù)文件的 link/script 標(biāo)簽插入到代碼畫板的 HTML 中,但是前端框架真的是太多了,又不能一個(gè)個(gè)去扣來(lái)寫死到頁(yè)面,就算是寫死了隨著框架版本的升級(jí),可能就無(wú)法滿足我們的需求。
以前寫頁(yè)面時(shí)經(jīng)常會(huì)用到 bootcdn,無(wú)意中發(fā)現(xiàn)它提供了相關(guān) API,可以直接拿來(lái)使用。接下來(lái)就得想辦法讓用戶通過(guò)界面選擇即可。
這個(gè) API 有三層數(shù)據(jù)結(jié)構(gòu):庫(kù) - 版本 - 資源鏈接。這個(gè)功能要用界面來(lái)實(shí)現(xiàn)肯定會(huì)非常臃腫,界面上可能會(huì)放很多按鈕。這就違背了「更簡(jiǎn)潔」的需求目標(biāo)。
這時(shí)就得參考下我們經(jīng)常使用的一些軟件是如何解決?簡(jiǎn)潔性?和 功能性 需求之間的矛盾問(wèn)題的,我比較喜歡 Sublime Text 的一些界面設(shè)計(jì),Command Palette 是我經(jīng)常使用的,所以我決定再模擬一個(gè)?Command Palette 來(lái)實(shí)現(xiàn)插入第三方庫(kù)的需求。而且重要的是這個(gè)?Command Palette 并不一定只用來(lái)實(shí)現(xiàn)這一個(gè)功能,或者后期會(huì)有一些別的功能需要添加,那這個(gè)?Command Palette 也是個(gè)很好的入口。
使用 electron 實(shí)現(xiàn)桌面應(yīng)用實(shí)現(xiàn)離線可用很多方法,比如使用 PWA 技術(shù)。但是 PWA 并不能給我?guī)?lái)一種原生應(yīng)用的那種可靠感,相反 electron?剛好可以解決我的顧慮。同時(shí)它可以把你的應(yīng)用打包成各個(gè)平臺(tái)(macOS/Window/Linux)的原生應(yīng)用。唯一的缺點(diǎn)就是安裝包確實(shí)很大,一般來(lái)講一個(gè) electron 應(yīng)用 安裝完 至少要 100 多兆,不過(guò)我覺得還能接受,畢竟硬盤存儲(chǔ)現(xiàn)在已經(jīng)很廉價(jià)了。
有人可能對(duì) electron 有抗拒,覺得 electron 應(yīng)用太龐大、占系統(tǒng)資源什么的,不過(guò)我們做的這個(gè)應(yīng)用并不需要常駐系統(tǒng),臨時(shí)性的使用一下,用完就關(guān)閉,正常寫生產(chǎn)環(huán)境的代碼肯定還是要換回 編輯器/IDE 的。同時(shí)因?yàn)?electron 降低了寫桌面應(yīng)用的門檻,確實(shí)有很多人把一個(gè)完整的在線的網(wǎng)頁(yè)直接嵌進(jìn)去,這也是有問(wèn)題的。
electron 還有一個(gè)好處,因?yàn)樗耆?HTML/CSS/JS 來(lái)實(shí)現(xiàn) UI(可以使用 Chrome only 的一些新功能),那我們理論上可以在做桌面應(yīng)用時(shí)順手把 web 應(yīng)用也做了。這就可以同時(shí)支持各個(gè)系統(tǒng)下的原生應(yīng)用,并且有 web 在線版本。如果你不愿意使用原生應(yīng)用,直接登錄 web.code-sketch.com 使用在線版也沒是一種選擇。這樣就使得我們的應(yīng)用具有真正的?跨端?能力。
由于我們團(tuán)隊(duì)都使用了 macbook,所以我優(yōu)先支持 macOS 的開發(fā),另外 macOS Mojave 的系統(tǒng)級(jí)別的暗色主題我也比較喜歡,剛好實(shí)現(xiàn)支持 mojave 暗色主題這個(gè)需求也做上。
三、框架的選擇大方向確定了,像框架選擇這個(gè)就簡(jiǎn)單了,基于 electron 的應(yīng)用,需要你區(qū)分開 render/main process 來(lái)選擇。
Render process渲染進(jìn)程?就是 electron 中界面的實(shí)現(xiàn)部分 ,一般來(lái)說(shuō)就是一個(gè) webview,選自己喜歡的框架即可。我使用 React 來(lái)實(shí)現(xiàn)界面。樣式方面就不再使用框架了,因?yàn)槲覀兊慕缑嬖瓌t上沒有復(fù)雜的元素,直接手寫 CSS,300 行內(nèi)基本上就可以解決問(wèn)題??赡苡腥藭?huì)覺得這不可能,實(shí)際情況是當(dāng)你寫樣式只跑在 Chrome 里面的時(shí)候那感覺完全爽到飛起,CSS variable/flex/grid/calc/vh/rem 什么的都可以拿來(lái)用,實(shí)現(xiàn)一個(gè)功能的成本就降低了很多。
我使用 Codemirror 來(lái)做為主界面的代碼編輯器,Monaco 也是一個(gè)好選擇,但是它有點(diǎn)過(guò)于龐大了,而且如果想要自定義功能得自己寫很多實(shí)現(xiàn)
主界面上的分割組件,使用了 React-split。
Main process主進(jìn)程 就是 electron 應(yīng)用程序的進(jìn)程,主要的區(qū)別在于主進(jìn)程中可以調(diào)用一些與原生操作系統(tǒng)交互的?API,比如對(duì)話框、系統(tǒng)風(fēng)格主題等。并且有 node 的運(yùn)行時(shí),可以引用 NPM 包。當(dāng)然渲染進(jìn)程也可以有 node 支持,但是我建議渲染進(jìn)程中就只放一些純前端的邏輯,這樣的話方便后期把應(yīng)用分離成 web 版
因?yàn)槲覀円?Sass 編譯功能,如果你也經(jīng)歷過(guò) node-sass 的各種問(wèn)題,那就應(yīng)該果斷選擇 dart-sass?— 使用 dart 實(shí)現(xiàn),編譯成了原生的 JS,沒有依賴問(wèn)題。dart-sass 我放在了 main process 中,因?yàn)槲以囘^(guò)放在 render process 中會(huì)有各種報(bào)錯(cuò)。如果 web 端要實(shí)現(xiàn)這個(gè)功能就需要其它的解決辦法了,比如做成一個(gè) http 服務(wù),讓 web 調(diào) http 服務(wù)。
Babel 的話我是放在了?渲染進(jìn)程?中以 script 標(biāo)簽的方式調(diào)用,這樣即使在 web 端 Babel 編譯也是可用的。
總之如果你使用 electron 構(gòu)建應(yīng)用并且引入的第三方 NPM 包可以?支持?運(yùn)行在客戶端(瀏覽器)上,那就盡量把包放在渲染進(jìn)程里面。
構(gòu)建工具我使用 Parcel 來(lái)構(gòu)建 React 而不是 Create React App。后者用來(lái)寫個(gè)小應(yīng)用還可以,稍微大一點(diǎn)的,需要定制化一些東西你就得 eject 出來(lái)一大堆 webpack 配置文件,即便是我已經(jīng)用 webpack 開發(fā)過(guò)幾個(gè)項(xiàng)目了,但是說(shuō)實(shí)話我還是沒用會(huì) webpack。寫 webpack 配置的時(shí)間足夠我自己寫 npm script 來(lái)滿足自己的需求了。
原生應(yīng)用打使用?electron-builder 來(lái)打包到平臺(tái)原生應(yīng)用,并且如果你有 Apple 開發(fā)者賬號(hào)的話應(yīng)用還可以提交到 AppStore 上去。
我目前的打包參數(shù)是這么配置的:
{ "build": { "productName": "Code Sketch", "extends": null, "directories": { "output": "release" }, "files": [ "icon.icns", "main.js", "src/*.js", "所有需要的文件", "package.json", "node_modules/@babel", "node_modules/sass" ], "mac": { "icon": "icon.icns", "category": "public.app-category.productivity", "target": [ "dmg" ] } } }
在你的 package.json 中添加 build 字段,productName, directories 這些按自己需要更改即可
四、分離開發(fā)環(huán)境 區(qū)分開開發(fā)環(huán)境代碼畫板項(xiàng)目開過(guò)過(guò)程中涉及兩個(gè)關(guān)鍵環(huán)境
Parcel 構(gòu)建環(huán)境(渲染進(jìn)程):Parcel 可以為你提供一些現(xiàn)在 JS 的轉(zhuǎn)譯工作,因此你可以放心使用例如 ES6 的 JS 新特性
Node.JS 運(yùn)行環(huán)境(主進(jìn)程+渲染進(jìn)程):這個(gè)取決于你的 electron 版本中集成的是 node 版本,比如:Node 10 中就沒有 ES Module,這意味著你如果要在 electron 主進(jìn)程?是無(wú)法識(shí)別 import 這樣的語(yǔ)句的,但是渲染進(jìn)程由于你使用了 Parcel 編譯,則無(wú)需考慮
這里溫馨提示下:想要做到 electron 中的 渲染進(jìn)程與主進(jìn)程之間共享 JS 代碼是非常困難的。就算是有辦法也會(huì)特別的別扭,我的建議是盡量分離這兩個(gè)進(jìn)程中的代碼,主進(jìn)程主要做一些系統(tǒng)級(jí)別的 API 調(diào)用、事件分發(fā)等,業(yè)務(wù)邏輯盡量放在渲染進(jìn)程中去做
如果非要共享,那建議多帶帶做成一個(gè) NPM 包分別做為主進(jìn)程運(yùn)行時(shí)依賴,和渲染進(jìn)程的 Parcel 編譯依賴,唯一的缺點(diǎn)就是實(shí)際上共享的代碼會(huì)有兩份。
渲染進(jìn)程中調(diào)用 node API 可能會(huì)和 Parcel 打包工具沖突,一般在調(diào)用比如文件模塊時(shí),可以加上 window.require(‘fs’) 這樣就可以兼容兩個(gè)環(huán)境:
get ipc() { if (window.require) { return window.require("electron").ipcRenderer } else { return { on() {}, send() {}, sendToHost() {} } } } this.ipc.send("event", data)
這樣的話你在瀏覽器端調(diào)試也不會(huì)產(chǎn)生報(bào)錯(cuò)。一般情況下,建議當(dāng)你用渲染進(jìn)程中的 JS 引用(require)包的時(shí)候都加上 window. 前綴就可以了。因?yàn)殇秩具M(jìn)程中 window 是全局變量,調(diào)用 require?和調(diào)用 window.require 是等價(jià)的
開發(fā)流程通常在測(cè)試的時(shí)候應(yīng)用會(huì)調(diào)用一些?electron 內(nèi)置的系統(tǒng)級(jí)別 API,這部分調(diào)用通常需要啟動(dòng) electron,但是有時(shí)候只有渲染進(jìn)程中 UI 界面上的改動(dòng),就不用再啟動(dòng) electron 了,直接在瀏覽器里面測(cè)試即可。使用 Parcel 運(yùn)行一個(gè)本地的服務(wù),這樣就可以在瀏覽器里面調(diào)試頁(yè)面。整個(gè)開發(fā)過(guò)程需要兩個(gè)命令(NPM Script):
啟動(dòng) Parcel 編譯服務(wù)器
"scripts": { "start": "./node_modules/.bin/parcel index.html -p 2044" }
調(diào)試 electron 原生功能,注意設(shè)置?ELECTRON_START_URL
"scripts": { "dev": "ELECTRON_START_URL=http://localhost:2044 yarn electron", }技術(shù)難點(diǎn)
整個(gè)應(yīng)用只有兩個(gè)功能是需要我們自己寫代碼實(shí)現(xiàn)的:日志控制臺(tái),Sublime 命令行。我們分別來(lái)分析下這兩個(gè)模塊的難點(diǎn)。
日志控制臺(tái) 的難點(diǎn)在于,我們需要打印任意類型的 JS 值。如果你對(duì) JS 了解比較多的話自然會(huì)想到在 JS 中所有的東西都是 對(duì)象,即 Object,那么實(shí)際上當(dāng)你想打印一個(gè)變量的時(shí)候,其實(shí)你只要把整個(gè) Object 遞歸的遍歷出來(lái),然后做成一個(gè)無(wú)限級(jí)的下拉菜單就可以了。看起來(lái)大概想下面這樣:
Sublime 命令行?實(shí)際上開發(fā)起來(lái)還是比較簡(jiǎn)單的,使用 React 很簡(jiǎn)單就實(shí)現(xiàn)了功能,比較麻煩的是調(diào)用 bootcdn 的接口,過(guò)程中我發(fā)現(xiàn)接口返回?cái)?shù)據(jù)量還是挺大的,有必要做上一層 localStorage 緩存,加快二次打開速度。
然而在使用的過(guò)程中你會(huì)發(fā)現(xiàn)當(dāng)我想插入一個(gè)前端庫(kù)需要很多操作,因?yàn)橛?三級(jí)選擇:庫(kù)-版本-CDN 鏈接。雖然這個(gè)流程解決了?所有用戶 的使用問(wèn)題,但是卻損害了 大部分 用戶的體驗(yàn)。這個(gè)時(shí)候插入一個(gè)常用庫(kù)的成本就很高了,所以我們就要加上一些快捷入口,來(lái)實(shí)現(xiàn)一鍵插入流行框架。
我們寫代碼的思路是滿足所有用戶的使用需求,但是一個(gè)好產(chǎn)品的思路是先滿足大多數(shù)用戶(80%)的常規(guī)需求,再讓其余的用戶(20%)可以有選擇
還有一個(gè)問(wèn)題比較典型就是 React 這類框架在渲染大列表并且進(jìn)行過(guò)濾(關(guān)鍵字查詢)時(shí)性能的問(wèn)題。注意這個(gè)性能問(wèn)題 并不是 引入框架產(chǎn)生的,真正的原因是當(dāng)你渲染的 HTML 節(jié)點(diǎn)數(shù)以千計(jì)的時(shí)候,批量操作 DOM 會(huì)使得 DOM Render 特別慢。
所以說(shuō)當(dāng)我們遇到性能問(wèn)題的時(shí)候應(yīng)該去查找問(wèn)題的根源,而不是停留在框架使用上,實(shí)際上在 DOM 操作這個(gè)層面來(lái)講 jQuery 提供了更多的性能優(yōu)化,比如自身的緩存系統(tǒng),以致于當(dāng)你在使用的時(shí)候很難發(fā)現(xiàn)有性能問(wèn)題。但是在類 React 框架中它們框架本身的重點(diǎn)并不在于解決你應(yīng)用的性能問(wèn)題。
類似我們上面講到的,實(shí)際上 jQuery 幫助你屏蔽了很多舞臺(tái)背后的東西,以致于你可以不用操心技術(shù)細(xì)節(jié),你甚至可以把 jQuery 當(dāng)做一個(gè) 產(chǎn)品 來(lái)使用,而類 React 框架你卻要親力親為的用他來(lái)設(shè)計(jì)你的代碼。
話題再轉(zhuǎn)回性能問(wèn)題。這時(shí)候需要我們?nèi)?shí)現(xiàn)一個(gè)類似于 react-window? 的功能,讓列表元素根據(jù)滾動(dòng)按需加載。這可能是一種通用的解決大列表加載的方案,但是我的解決方法更粗暴,因?yàn)槲覀兊南吕^(guò)濾功能使用時(shí)用戶只關(guān)注?最佳的匹配項(xiàng)?即可,后面匹配程度不高的項(xiàng)可以直接限制數(shù)量裁剪就行了嘛。很少有用戶會(huì)一直滾動(dòng)到下面去查找某個(gè)選項(xiàng),如果有,那就說(shuō)明我們這個(gè)匹配做的有問(wèn)題。
slice() { const idx = (this.props.itemsPerPage || 50) * (this.state.activeFrame + 1) return this.props.items.slice(0, idx) }
整個(gè)匹配篩選的狀態(tài)大概是這樣的:
this.state = { // 當(dāng)前第N步選擇 step: 0, // 當(dāng)前步驟數(shù)據(jù) items: [], // 是否顯示 active: false, // 當(dāng)前選中項(xiàng) current: {}, // 過(guò)濾關(guān)鍵字 keyword: "" }
這個(gè)?items?是當(dāng)前步驟的所有數(shù)據(jù),實(shí)際上我們這個(gè)組件是支持無(wú)限級(jí)的擴(kuò)展的,那么我們通過(guò)組件的 props 傳入所有層級(jí)的數(shù)據(jù),然后持久存儲(chǔ)在內(nèi)存中。這個(gè) 所有層級(jí)的數(shù)據(jù) 是數(shù)據(jù)結(jié)構(gòu)層面的,實(shí)際上它可能是通過(guò)異步接口獲取的。
再來(lái)看看我們組件提供的所有 props:
static defaultProps = { step: 0, active: false, data: [[]], // 無(wú)限層級(jí)數(shù)據(jù) [[], [], [], ...] // 數(shù)據(jù)的主鍵,用于鉤子函數(shù)返回用戶選擇的結(jié)果集 pk: "id", autoFocus: true, activeCls: "active", delay: 300, defaultSelected: 0, placeholder: "", async: false, alias: [], done: () => {} }
這些數(shù)據(jù)都可以通過(guò)組件的 props 傳入,這就意味著我們的這個(gè)組件才是真正的組件,別人也可以使用這樣的功能,而他們并不用在意里面的細(xì)節(jié),使用者只需要做好類似調(diào)用自己接口的這種業(yè)務(wù)邏輯。
組件的調(diào)用大概是這樣的:
async?這個(gè) props 實(shí)際上是一個(gè)異步調(diào)用的鉤子方法,它會(huì)回傳給你組件上當(dāng)前操作的相關(guān)數(shù)據(jù)狀態(tài),通過(guò)這些數(shù)據(jù)使用者就可以按自己的需求在不同的步驟上調(diào)用不同的方法
export const injectData = (step, item, results, cb) => { const API = "https://api.bootcdn.cn/libraries" if (step === 0) { fetchData(`${API}.min.json`) .then(processLibraryData) .then(cb) } else if (step === 1) { // ... } else if (step === 2) { // ... } }另外關(guān)于 React 這里安利下自己翻譯過(guò)的一個(gè)教程:React 模式,里面講到 18 種短小精悍的 React 模式案例,非常簡(jiǎn)單易懂。
還有一個(gè)小竅門,我們?cè)谶m配暗色主題時(shí),傳統(tǒng)的方法是直接寫兩套主題 CSS 代碼,實(shí)際上我們要使用 CSS Variable 的話完全沒必要生成兩套了,背景色,字體都做成 CSS 變量,切換的時(shí)候只需要?jiǎng)討B(tài)往頁(yè)面插入更新過(guò)的 CSS 變量值即可
系統(tǒng)的一些參數(shù)想直接傳給渲染進(jìn)程也是比較麻煩的,我的做法是直接從主進(jìn)程中的 loadUrl 方法上以 queryString 的方式傳到渲染頁(yè)面的 URL 上
const query = { theme: osTheme, app_path: app.getAppPath(), home_dir: app.getPath("home") } mainWindow.loadURL(process.env.ELECTRON_START_URL ? url.format({ slashes: true, protocol: "http:", hostname: "localhost", port: 2044, query }) : url.format({ slashes: true, protocol: "file:", pathname: path.resolve(app.getAppPath(), "./dist/index.html"), query }))像程序運(yùn)行時(shí)的一些參數(shù)(比如程序的根目錄)也可以這么動(dòng)態(tài)傳過(guò)去,而且還有一個(gè)好處就是你甚至可以在渲染進(jìn)程中測(cè)試與這些參數(shù)相關(guān)的功能。
五、宣傳 demo 視頻錄制我會(huì)把最終所有功能的使用方法錄制成一個(gè)視頻,萬(wàn)一有人不不想下載你的軟件,只是要了解一下,這就是個(gè)很好的方法。我同時(shí)上傳到了 Youtube 和 bilibili 這兩個(gè)平臺(tái),其它的都有廣告就沒必要了
使用 Quicktime Player 即可,錄制完使用 iMovie 轉(zhuǎn)碼成兩倍速率的 mp4。如果你有興趣還可以加上一段音樂(lè)什么的,讓視頻看起來(lái)更靈動(dòng)
域名申請(qǐng)域名是一個(gè)能讓用戶記住你產(chǎn)品的方法,如果你做的是一個(gè)成型的產(chǎn)品,那就一定要申請(qǐng)個(gè)域名。
我總是有這樣的體驗(yàn),有的時(shí)候看到一個(gè)非常不錯(cuò)的產(chǎn)品但由于當(dāng)時(shí)沒需求就忽略了,想起來(lái)或者突然有需求的時(shí)候缺記不起來(lái)名字叫什么了。
事實(shí)上代碼畫板最開始我給他起的名字是 code playground,這個(gè)更直觀,但是名字太長(zhǎng),而且想用到的一些域名呀、Github 名、NPM 包都被注冊(cè)了。
想來(lái)想去就換成了 code sketch,這和符合我們的設(shè)計(jì)初衷,即:一邊是代碼,一邊是效果/草圖
域名申請(qǐng)我一般會(huì)上 Godaddy,不用備案,.com 域名一年 ¥65.00,然后 DNS 服務(wù)器轉(zhuǎn)到了?cloudflare,后續(xù)域名也會(huì)直接轉(zhuǎn)到?cloudflare。因?yàn)閾?jù)說(shuō)以后在 cloudflare 上續(xù)費(fèi)域名最便宜
網(wǎng)站搭建宣傳網(wǎng)站直接放在 github pages 上,做個(gè)自定義域即可,實(shí)在是太方便了。而且還有 SSL 支持,Github 真的是業(yè)界良心
web 版的代碼畫板,由于我們把渲染進(jìn)程中的代碼分離開發(fā),所以直接把 parcel 打包出來(lái)的靜態(tài)文件也做成 github pages 就可以了,爽歪歪,網(wǎng)站就等于一分錢不花了。后續(xù)做一些 web 版的增強(qiáng)功能時(shí),可以做成前后端分離的 http 服務(wù),這就是后話了
加入 Google?analytics 代碼GA 可以讓你了解網(wǎng)站的用戶分布情況,清楚的知道網(wǎng)站訪問(wèn)的波動(dòng)。比如說(shuō)你把自己的鏈接放到某個(gè)網(wǎng)站上分享了,GA 里面就能看出來(lái)所有的推薦來(lái)源和波動(dòng),對(duì)于運(yùn)營(yíng)來(lái)說(shuō)是非常有必要的
廣告語(yǔ)這個(gè)我還真想了好長(zhǎng)時(shí)間,基于我對(duì)于代碼畫板的定義,我覺得它應(yīng)該是一個(gè)我們有一個(gè)想法的時(shí)候需要快速去實(shí)現(xiàn)一個(gè) demo 的地方,想來(lái)想去就定了一段看起來(lái)文鄒鄒的話,雖然聽名字根本不知道它是干啥用的,但是沒關(guān)系,程序員寫東西就是要有個(gè)性,因?yàn)槲业氖鼙娭挥凶约骸?/p>
First place where the code was written...六、匯總使用到的庫(kù)與工具
一個(gè)你最初寫代碼的地方...麻雀雖小,五臟俱全。我們來(lái)看下代碼畫板總共用到了多少東西:
框架/庫(kù)
electronjs
react
babeljs
NPM 模塊
codemirror 及其插件
react-split
sublime-command-palette
打包/工具
parceljs
electron-builder
bootcdn
設(shè)計(jì)與素材
sketch
Free AppIcon Generator
Inconsolata 字體
Gallary CSS?純 CSS 實(shí)現(xiàn)的焦點(diǎn)圖,用于宣傳頁(yè)展示
七、結(jié)語(yǔ)總結(jié)實(shí)事上我自己的開發(fā)這個(gè)應(yīng)用的時(shí)候并沒有嚴(yán)格按照這篇文章的順序執(zhí)行,而是想到一些實(shí)現(xiàn)一些,可能一個(gè)功能實(shí)現(xiàn)了后來(lái)覺得不好又干掉了,是不斷的取舍、提煉的結(jié)果。
開發(fā)中我也不斷的問(wèn)自己這個(gè)功能是否有必要,如果可有可無(wú)那是不是可以去掉,這樣才能使得用戶更加關(guān)注于代碼本身。
整個(gè)開發(fā)過(guò)程中自己實(shí)現(xiàn)的功能模塊并不多,只有控制臺(tái)、命令行窗口是自己實(shí)現(xiàn)的,其它的功能基本上都是靠社區(qū)現(xiàn)有的工具庫(kù)來(lái)完成的,從這一點(diǎn)來(lái)說(shuō)前端技術(shù)的生態(tài)還是挺好的。這使得當(dāng)我從整體上構(gòu)思一個(gè)產(chǎn)品時(shí)我不必在意那些細(xì)節(jié),雖然過(guò)程中還是能感覺到前端工具/庫(kù)的割裂感,但是整體而言還是向好的,畢竟工具對(duì)于開發(fā)者只是一種選擇的。
八、引用
https://github.com/keelii/code-sketch
http://www.tweaknow.com/appicongenerator.php
http://benschwarz.github.io/gallery-css/
https://addyosmani.com/blog/react-window/
https://github.com/keelii/reactpatterns.cn
原文:https://mp.weixin.qq.com/s/Lbnx0aYdRa5iDhOkCgEWjg
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/102510.html
摘要:前陣子,有些師弟師妹問(wèn)我在淘寶,前端開發(fā)主要是在做什么事情作為一個(gè)在淘寶已經(jīng)工作年的老兵,我想我有資格來(lái)全面地回答一下這個(gè)問(wèn)題,并通過(guò)這個(gè)機(jī)會(huì)向外部介紹一下我們團(tuán)隊(duì)的同學(xué)。以上便是我們?cè)谔詫氈饕谧龅氖虑椤? 前陣子,有些師弟師妹問(wèn)我:在淘寶,前端開發(fā)主要是在做什么事情? 作為一個(gè)在淘寶已經(jīng)工作 5 年的「老兵」,我想我有資格來(lái)全面地回答一下這個(gè)問(wèn)題,并通過(guò)這個(gè)機(jī)會(huì)向外部介紹一下我們團(tuán)隊(duì)...
摘要:所以在小程序出現(xiàn)之后,一股框架之風(fēng)也很快的出現(xiàn),微信小程序剛推出之后,就出現(xiàn)了兩個(gè)比較出名的小程序開發(fā)框架,。 原文地址:https://ant-move.github.io/we... 這里說(shuō)的去除小程序框架其實(shí)并不嚴(yán)謹(jǐn),因?yàn)樾〕绦虮旧硪菜闶且粋€(gè)框架,而且是一個(gè)功能更加完善的框架系統(tǒng)。在前端的概念中,我們一般說(shuō)一個(gè)框架是指一個(gè)用來(lái)幫助開發(fā)者構(gòu)建用戶界面的框架,而小程序框架本身不僅僅包...
摘要:跨端框架壹個(gè)理想主義團(tuán)隊(duì)的開源作品歷經(jīng)近個(gè)月打磨,滴滴跨端方案終于開源了真正專注于一套代碼運(yùn)行多端。這時(shí)候我們專門成立了一個(gè)人的小項(xiàng)目組,完成一個(gè)名為的項(xiàng)目,一期目標(biāo)是不影響用戶發(fā)揮,不依賴框架方的原則性實(shí)現(xiàn)一套代碼運(yùn)行和微信小程序。 Chameleon跨端框架——壹個(gè)理想主義團(tuán)隊(duì)的開源作品 歷經(jīng)近20個(gè)月打磨,滴滴跨端方案chameleon終于開源了https://github.co...
摘要:前端每周清單專注前端領(lǐng)域內(nèi)容,以對(duì)外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn)分為新聞熱點(diǎn)開發(fā)教程工程實(shí)踐深度閱讀開源項(xiàng)目巔峰人生等欄目。 前端每周清單專注前端領(lǐng)域內(nèi)容,以對(duì)外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn);分為新聞熱點(diǎn)、開發(fā)教程、工程實(shí)踐、深度閱讀、開源項(xiàng)目、巔峰人生等欄目。歡迎關(guān)注【前端之巔】微信公眾號(hào)(ID:frontshow),及時(shí)獲取前端每周清單;本文則是對(duì)于...
閱讀 967·2021-11-24 09:39
閱讀 3396·2021-10-27 14:20
閱讀 2326·2019-08-30 14:08
閱讀 3368·2019-08-29 16:34
閱讀 2182·2019-08-26 12:14
閱讀 2110·2019-08-26 11:54
閱讀 2779·2019-08-26 11:44
閱讀 2480·2019-08-26 11:38