成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

如何使用前端技術(shù)開發(fā)一個(gè)桌面跨端應(yīng)用

shixinzhang / 2767人閱讀

摘要:使用實(shí)現(xiàn)桌面應(yīng)用實(shí)現(xiàn)離線可用很多方法,比如使用技術(shù)。還有一個(gè)好處,因?yàn)樗耆趤韺?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),如果你只想知道一些拿來即用的「干貨」,或許這篇文章并不是一個(gè)好的選擇

一、定位需求

事情的起因是這樣的,因?yàn)槲覀儍?nèi)部會(huì)有一些培訓(xùn)會(huì)議。會(huì)經(jīng)?,F(xiàn)場(chǎng)演示一些代碼片段。比如說我們講到 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è)拿來即用的工具,它不應(yīng)該有更復(fù)雜的功能,比如用戶登錄、代碼片段的管理等。這些需求不是它要解決的。代碼畫板會(huì)提供一個(gè)簡(jiǎn)單的導(dǎo)出成 HTML 文件的功能,可以方便用戶存儲(chǔ)整個(gè) HTML 文件。

既然是用來演示代碼的,那么它的界面上應(yīng)該只有兩個(gè)東西,一個(gè)是 代碼,一個(gè)就是?預(yù)覽。像代碼/控制臺(tái)切換的功能都做成 tab 的形式,正常情況不需要讓他們展示出來。像 codepen 那樣把所有的代碼編輯器功能都展示出來我認(rèn)為是不對(duì)的。

codepen 的界面給人感覺非常復(fù)雜,有很多功能點(diǎn)。當(dāng)然我并不是在批評(píng)它,codepen 做為一個(gè)需要商業(yè)化運(yùn)營(yíng)的軟件,勢(shì)必會(huì)做的非常復(fù)雜,這樣才能滿足更多用戶的需求。然而程序員寫軟件則可以完全按照自己的想法來,哪怕這個(gè)應(yīng)用只給自己一個(gè)人用呢。

桌面應(yīng)用的設(shè)計(jì)

桌面應(yīng)用的設(shè)計(jì)和 web 界面的設(shè)計(jì)還是有些細(xì)微區(qū)別的,同樣的基于 electron 的應(yīng)用,有的應(yīng)用會(huì)讓人感覺很「原生」,有的則一眼就能看出來是用 CSS 畫的。我在設(shè)計(jì)代碼畫板的時(shí)候也盡量向原生靠近,避免產(chǎn)生落差感。比如禁用鼠標(biāo)手型圖標(biāo)、在按鈕或者非可選元素上禁止用戶選擇:

cursor: default;
user-select: none

因?yàn)閷?shí)際上用戶在使用一款應(yīng)用的時(shí)候感性的因素影響占很大一部分,比如說有人不喜歡 electron 可能就是因?yàn)榭吹竭^ electron 里面嵌一個(gè)完整的 web 頁面的操作,這就讓人很反感。但是這不是 electron 的問題,而是應(yīng)用設(shè)計(jì)者的問題。

應(yīng)用標(biāo)識(shí)的設(shè)計(jì)

說實(shí)話應(yīng)用 logo 設(shè)計(jì)我也是業(yè)余水平,但是聊勝于無。既然水平不行,那就盡量設(shè)計(jì)的不難看就行了。可以參考一些好的設(shè)計(jì)。我用 sketch 畫出 logo 的外形,sketch 有很多 macOS 的模塊可以從網(wǎng)上下載下來,直接基于模板修改就可以了。

代碼畫板主要的界面是分割開的兩個(gè)面板,左邊是代碼,右邊是預(yù)覽。所以我就大概畫了一個(gè)形狀

這個(gè) logo 有個(gè)問題就是線條過多,小尺寸的時(shí)候看不清楚。這個(gè)問題我暫時(shí)先忽略了,畢竟我還不是專業(yè)的,后續(xù)有好的創(chuàng)意可以再改

默認(rèn)設(shè)置

代碼畫板也 不會(huì)有 設(shè)置界面,因?yàn)槌S玫脑O(shè)置都預(yù)定義好了,你不需要配置。頂多改變下代碼字體的大小。使用編輯器的通用快捷鍵 command++/-?就解決了,或者插入三方庫,直接使用編輯器的通用命令快捷鍵 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),萬一還是有人不會(huì)用的話,可以去看幫助文檔。里面有所有的快捷鍵。

界面中間的分割條可以自定義拖動(dòng),雙擊重置平分界面

剛開始的時(shí)候我把每個(gè) tab 頁簽都分割成多帶帶的面板,因?yàn)槲矣X得這個(gè)能拖動(dòng)自定義面板大小的交互實(shí)在是太爽了,忍不住想去拖動(dòng)它。但是后來想想,其實(shí)并沒有必要,我們寫代碼時(shí)應(yīng)該更專注于代碼本身,如果只有兩個(gè)面板,那么這個(gè)界面無論是認(rèn)知還是使用起來就沒有任何困難。

因?yàn)槲覀儾⒉恍枰岩欢训墓δ艿慕缑嫠そo用戶,讓他們自己去選擇。

三、技術(shù)調(diào)研 實(shí)現(xiàn)控制臺(tái)

通過使用流行的幾款在線代碼運(yùn)行工具,我發(fā)現(xiàn)他們有一個(gè)共同的問題:控制臺(tái)很難用。無法像 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 的日志展示界面(無限級(jí)聯(lián)選擇)

日志界面應(yīng)該有下面這些功能:

展示任意 JS 類型的數(shù)據(jù)

Primitive 類型的數(shù)據(jù)顯示不同的顏色(number - 藍(lán)色,string - 綠色)

Object 類型默認(rèn)折疊起來,點(diǎn)擊按鈕展示子級(jí),屬性過多需要展示縮略信息

數(shù)組前應(yīng)該有長(zhǎng)度標(biāo)記

能展示 JS 運(yùn)行時(shí)的報(bào)錯(cuò) Error 信息

集成現(xiàn)代化的前端框工作流

現(xiàn)代化的前端寫頁面肯定不是 HTML/CSS/JS 一把梭了,至少應(yīng)該有 Sass/Babel 的支持吧。

Sass 嵌套能讓你少寫很多選擇器,當(dāng)然 Less 也可以,但是在我們的這個(gè)應(yīng)用里面區(qū)別不大,一般來說臨時(shí)性的寫一些代碼很少會(huì)用到它們的細(xì)節(jié)功能。有?變量?和?選擇器?嵌套就夠了

Babel 主要是解決了寫 React 的問題,不用再安裝一大堆的構(gòu)建工具了,直接使用 UMDReact/ReactDOM 就可以了,而且 electron 內(nèi)嵌的 chromium 也支持了 es6 的 class 寫法,實(shí)際上 Babel 主要的目的還是用來轉(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 頁面時(shí)通常是要引用很多第三方類庫的,比如:Bootstrp/jQuery 等。我希望有一種方法可以方便的引用到這些庫,直接把庫文件的 link/script 標(biāo)簽插入到代碼畫板的 HTML 中,但是前端框架真的是太多了,又不能一個(gè)個(gè)去扣來寫死到頁面,就算是寫死了隨著框架版本的升級(jí),可能就無法滿足我們的需求。

以前寫頁面時(shí)經(jīng)常會(huì)用到 bootcdn,無意中發(fā)現(xiàn)它提供了相關(guān) API,可以直接拿來使用。接下來就得想辦法讓用戶通過界面選擇即可。

這個(gè) API 有三層數(shù)據(jù)結(jié)構(gòu):庫 - 版本 - 資源鏈接。這個(gè)功能要用界面來實(shí)現(xiàn)肯定會(huì)非常臃腫,界面上可能會(huì)放很多按鈕。這就違背了「更簡(jiǎn)潔」的需求目標(biāo)。

這時(shí)就得參考下我們經(jīng)常使用的一些軟件是如何解決?簡(jiǎn)潔性?和 功能性 需求之間的矛盾問題的,我比較喜歡 Sublime Text 的一些界面設(shè)計(jì),Command Palette 是我經(jīng)常使用的,所以我決定再模擬一個(gè)?Command Palette 來實(shí)現(xiàn)插入第三方庫的需求。而且重要的是這個(gè)?Command Palette 并不一定只用來實(shí)現(xiàn)這一個(gè)功能,或者后期會(huì)有一些別的功能需要添加,那這個(gè)?Command Palette 也是個(gè)很好的入口。

使用 electron 實(shí)現(xiàn)桌面應(yīng)用

實(shí)現(xiàn)離線可用很多方法,比如使用 PWA 技術(shù)。但是 PWA 并不能給我?guī)硪环N原生應(yīng)用的那種可靠感,相反 electron?剛好可以解決我的顧慮。同時(shí)它可以把你的應(yīng)用打包成各個(gè)平臺(tái)(macOS/Window/Linux)的原生應(yīng)用。唯一的缺點(diǎn)就是安裝包確實(shí)很大,一般來講一個(gè) electron 應(yīng)用 安裝完 至少要 100 多兆,不過我覺得還能接受,畢竟硬盤存儲(chǔ)現(xiàn)在已經(jīng)很廉價(jià)了。

有人可能對(duì) electron 有抗拒,覺得 electron 應(yīng)用太龐大、占系統(tǒng)資源什么的,不過我們做的這個(gè)應(yīng)用并不需要常駐系統(tǒng),臨時(shí)性的使用一下,用完就關(guān)閉,正常寫生產(chǎn)環(huán)境的代碼肯定還是要換回 編輯器/IDE 的。同時(shí)因?yàn)?electron 降低了寫桌面應(yīng)用的門檻,確實(shí)有很多人把一個(gè)完整的在線的網(wǎng)頁直接嵌進(jìn)去,這也是有問題的。

electron 還有一個(gè)好處,因?yàn)樗耆?HTML/CSS/JS 來實(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 來選擇。

Render process

渲染進(jìn)程?就是 electron 中界面的實(shí)現(xiàn)部分 ,一般來說就是一個(gè) webview,選自己喜歡的框架即可。我使用 React 來實(shí)現(xiàn)界面。樣式方面就不再使用框架了,因?yàn)槲覀兊慕缑嬖瓌t上沒有復(fù)雜的元素,直接手寫 CSS,300 行內(nèi)基本上就可以解決問題??赡苡腥藭?huì)覺得這不可能,實(shí)際情況是當(dāng)你寫樣式只跑在 Chrome 里面的時(shí)候那感覺完全爽到飛起,CSS variable/flex/grid/calc/vh/rem 什么的都可以拿來用,實(shí)現(xiàn)一個(gè)功能的成本就降低了很多。

我使用 Codemirror 來做為主界面的代碼編輯器,Monaco 也是一個(gè)好選擇,但是它有點(diǎn)過于龐大了,而且如果想要自定義功能得自己寫很多實(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)歷過 node-sass 的各種問題,那就應(yīng)該果斷選擇 dart-sass?— 使用 dart 實(shí)現(xiàn),編譯成了原生的 JS,沒有依賴問題。dart-sass 我放在了 main process 中,因?yàn)槲以囘^放在 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 來構(gòu)建 React 而不是 Create React App。后者用來寫個(gè)小應(yīng)用還可以,稍微大一點(diǎn)的,需要定制化一些東西你就得 eject 出來一大堆 webpack 配置文件,即便是我已經(jīng)用 webpack 開發(fā)過幾個(gè)項(xiàng)目了,但是說實(shí)話我還是沒用會(huì) webpack。寫 webpack 配置的時(shí)間足夠我自己寫 npm script 來滿足自己的需求了。

原生應(yīng)用打

使用?electron-builder 來打包到平臺(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)目開過過程中涉及兩個(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)程?是無法識(shí)別 import 這樣的語句的,但是渲染進(jìn)程由于你使用了 Parcel 編譯,則無需考慮

這里溫馨提示下:想要做到 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)試頁面。整個(gè)開發(fā)過程需要兩個(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 命令行。我們分別來分析下這兩個(gè)模塊的難點(diǎn)。

日志控制臺(tái) 的難點(diǎn)在于,我們需要打印任意類型的 JS 值。如果你對(duì) JS 了解比較多的話自然會(huì)想到在 JS 中所有的東西都是 對(duì)象,即 Object,那么實(shí)際上當(dāng)你想打印一個(gè)變量的時(shí)候,其實(shí)你只要把整個(gè) Object 遞歸的遍歷出來,然后做成一個(gè)無限級(jí)的下拉菜單就可以了??雌饋泶蟾畔胂旅孢@樣:

Sublime 命令行?實(shí)際上開發(fā)起來還是比較簡(jiǎn)單的,使用 React 很簡(jiǎn)單就實(shí)現(xiàn)了功能,比較麻煩的是調(diào)用 bootcdn 的接口,過程中我發(fā)現(xiàn)接口返回?cái)?shù)據(jù)量還是挺大的,有必要做上一層 localStorage 緩存,加快二次打開速度。

然而在使用的過程中你會(huì)發(fā)現(xiàn)當(dāng)我想插入一個(gè)前端庫需要很多操作,因?yàn)橛?三級(jí)選擇:庫-版本-CDN 鏈接。雖然這個(gè)流程解決了?所有用戶 的使用問題,但是卻損害了 大部分 用戶的體驗(yàn)。這個(gè)時(shí)候插入一個(gè)常用庫的成本就很高了,所以我們就要加上一些快捷入口,來實(shí)現(xiàn)一鍵插入流行框架。

我們寫代碼的思路是滿足所有用戶的使用需求,但是一個(gè)好產(chǎn)品的思路是先滿足大多數(shù)用戶(80%)的常規(guī)需求,再讓其余的用戶(20%)可以有選擇

還有一個(gè)問題比較典型就是 React 這類框架在渲染大列表并且進(jìn)行過濾(關(guān)鍵字查詢)時(shí)性能的問題。注意這個(gè)性能問題 并不是 引入框架產(chǎn)生的,真正的原因是當(dāng)你渲染的 HTML 節(jié)點(diǎn)數(shù)以千計(jì)的時(shí)候,批量操作 DOM 會(huì)使得 DOM Render 特別慢。

所以說當(dāng)我們遇到性能問題的時(shí)候應(yīng)該去查找問題的根源,而不是停留在框架使用上,實(shí)際上在 DOM 操作這個(gè)層面來講 jQuery 提供了更多的性能優(yōu)化,比如自身的緩存系統(tǒng),以致于當(dāng)你在使用的時(shí)候很難發(fā)現(xiàn)有性能問題。但是在類 React 框架中它們框架本身的重點(diǎn)并不在于解決你應(yīng)用的性能問題。

類似我們上面講到的,實(shí)際上 jQuery 幫助你屏蔽了很多舞臺(tái)背后的東西,以致于你可以不用操心技術(shù)細(xì)節(jié),你甚至可以把 jQuery 當(dāng)做一個(gè) 產(chǎn)品 來使用,而類 React 框架你卻要親力親為的用他來設(shè)計(jì)你的代碼。

話題再轉(zhuǎn)回性能問題。這時(shí)候需要我們?nèi)?shí)現(xiàn)一個(gè)類似于 react-window? 的功能,讓列表元素根據(jù)滾動(dòng)按需加載。這可能是一種通用的解決大列表加載的方案,但是我的解決方法更粗暴,因?yàn)槲覀兊南吕^濾功能使用時(shí)用戶只關(guān)注?最佳的匹配項(xiàng)?即可,后面匹配程度不高的項(xiàng)可以直接限制數(shù)量裁剪就行了嘛。很少有用戶會(huì)一直滾動(dòng)到下面去查找某個(gè)選項(xiàng),如果有,那就說明我們這個(gè)匹配做的有問題。

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ān)鍵字
    keyword: ""
}

這個(gè)?items?是當(dāng)前步驟的所有數(shù)據(jù),實(shí)際上我們這個(gè)組件是支持無限級(jí)的擴(kuò)展的,那么我們通過組件的 props 傳入所有層級(jí)的數(shù)據(jù),然后持久存儲(chǔ)在內(nèi)存中。這個(gè) 所有層級(jí)的數(shù)據(jù) 是數(shù)據(jù)結(jié)構(gòu)層面的,實(shí)際上它可能是通過異步接口獲取的。

再來看看我們組件提供的所有 props

static defaultProps = {
    step: 0,
    active: false,
    data: [[]],    // 無限層級(jí)數(shù)據(jù) [[], [], [], ...]
    // 數(shù)據(jù)的主鍵,用于鉤子函數(shù)返回用戶選擇的結(jié)果集
    pk: "id",

    autoFocus: true,
    activeCls: "active",
    delay: 300,
    defaultSelected: 0,
    placeholder: "",
    async: false,
    alias: [],
    done: () => {}
}

這些數(shù)據(jù)都可以通過組件的 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),通過這些數(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 這里安利下自己翻譯過的一個(gè)教程:React 模式,里面講到 18 種短小精悍的 React 模式案例,非常簡(jiǎn)單易懂。

還有一個(gè)小竅門,我們?cè)谶m配暗色主題時(shí),傳統(tǒng)的方法是直接寫兩套主題 CSS 代碼,實(shí)際上我們要使用 CSS Variable 的話完全沒必要生成兩套了,背景色,字體都做成 CSS 變量,切換的時(shí)候只需要?jiǎng)討B(tài)往頁面插入更新過的 CSS 變量值即可

系統(tǒng)的一些參數(shù)想直接傳給渲染進(jìn)程也是比較麻煩的,我的做法是直接從主進(jìn)程中的 loadUrl 方法上以 queryString 的方式傳到渲染頁面的 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)傳過去,而且還有一個(gè)好處就是你甚至可以在渲染進(jìn)程中測(cè)試與這些參數(shù)相關(guān)的功能。

五、宣傳 demo 視頻錄制

我會(huì)把最終所有功能的使用方法錄制成一個(gè)視頻,萬一有人不不想下載你的軟件,只是要了解一下,這就是個(gè)很好的方法。我同時(shí)上傳到了 Youtube 和 bilibili 這兩個(gè)平臺(tái),其它的都有廣告就沒必要了

使用 Quicktime Player 即可,錄制完使用 iMovie 轉(zhuǎn)碼成兩倍速率的 mp4。如果你有興趣還可以加上一段音樂什么的,讓視頻看起來更靈動(dòng)

域名申請(qǐng)

域名是一個(gè)能讓用戶記住你產(chǎn)品的方法,如果你做的是一個(gè)成型的產(chǎn)品,那就一定要申請(qǐng)個(gè)域名。

我總是有這樣的體驗(yàn),有的時(shí)候看到一個(gè)非常不錯(cuò)的產(chǎn)品但由于當(dāng)時(shí)沒需求就忽略了,想起來或者突然有需求的時(shí)候缺記不起來名字叫什么了。

事實(shí)上代碼畫板最開始我給他起的名字是 code playground,這個(gè)更直觀,但是名字太長(zhǎng),而且想用到的一些域名呀、Github 名、NPM 包都被注冊(cè)了。

想來想去就換成了 code sketch,這和符合我們的設(shè)計(jì)初衷,即:一邊是代碼,一邊是效果/草圖

域名申請(qǐng)我一般會(huì)上 Godaddy,不用備案,.com 域名一年 ¥65.00,然后 DNS 服務(wù)器轉(zhuǎn)到了?cloudflare,后續(xù)域名也會(huì)直接轉(zhuǎn)到?cloudflare。因?yàn)閾?jù)說以后在 cloudflare 上續(xù)費(fèi)域名最便宜

網(wǎng)站搭建

宣傳網(wǎng)站直接放在 github pages 上,做個(gè)自定義域即可,實(shí)在是太方便了。而且還有 SSL 支持,Github 真的是業(yè)界良心

web 版的代碼畫板,由于我們把渲染進(jìn)程中的代碼分離開發(fā),所以直接把 parcel 打包出來的靜態(tài)文件也做成 github pages 就可以了,爽歪歪,網(wǎng)站就等于一分錢不花了。后續(xù)做一些 web 版的增強(qiáng)功能時(shí),可以做成前后端分離的 http 服務(wù),這就是后話了

加入 Google?analytics 代碼

GA 可以讓你了解網(wǎng)站的用戶分布情況,清楚的知道網(wǎng)站訪問的波動(dòng)。比如說你把自己的鏈接放到某個(gè)網(wǎng)站上分享了,GA 里面就能看出來所有的推薦來源和波動(dòng),對(duì)于運(yùn)營(yíng)來說是非常有必要的

廣告語

這個(gè)我還真想了好長(zhǎng)時(shí)間,基于我對(duì)于代碼畫板的定義,我覺得它應(yīng)該是一個(gè)我們有一個(gè)想法的時(shí)候需要快速去實(shí)現(xiàn)一個(gè) demo 的地方,想來想去就定了一段看起來文鄒鄒的話,雖然聽名字根本不知道它是干啥用的,但是沒關(guān)系,程序員寫東西就是要有個(gè)性,因?yàn)槲业氖鼙娭挥凶约骸?/p>

First place where the code was written...
一個(gè)你最初寫代碼的地方...
六、匯總使用到的庫與工具

麻雀雖小,五臟俱全。我們來看下代碼畫板總共用到了多少東西:

框架/庫

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)圖,用于宣傳頁展示

七、結(jié)語總結(jié)

實(shí)事上我自己的開發(fā)這個(gè)應(yīng)用的時(shí)候并沒有嚴(yán)格按照這篇文章的順序執(zhí)行,而是想到一些實(shí)現(xiàn)一些,可能一個(gè)功能實(shí)現(xiàn)了后來覺得不好又干掉了,是不斷的取舍、提煉的結(jié)果。

開發(fā)中我也不斷的問自己這個(gè)功能是否有必要,如果可有可無那是不是可以去掉,這樣才能使得用戶更加關(guān)注于代碼本身。

整個(gè)開發(fā)過程中自己實(shí)現(xiàn)的功能模塊并不多,只有控制臺(tái)、命令行窗口是自己實(shí)現(xiàn)的,其它的功能基本上都是靠社區(qū)現(xiàn)有的工具庫來完成的,從這一點(diǎn)來說前端技術(shù)的生態(tài)還是挺好的。這使得當(dāng)我從整體上構(gòu)思一個(gè)產(chǎn)品時(shí)我不必在意那些細(xì)節(jié),雖然過程中還是能感覺到前端工具/庫的割裂感,但是整體而言還是向好的,畢竟工具對(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://systransis.cn/yun/102510.html

相關(guān)文章

  • 前端開發(fā)在淘寶主要是在做什么事情?

    摘要:前陣子,有些師弟師妹問我在淘寶,前端開發(fā)主要是在做什么事情作為一個(gè)在淘寶已經(jīng)工作年的老兵,我想我有資格來全面地回答一下這個(gè)問題,并通過這個(gè)機(jī)會(huì)向外部介紹一下我們團(tuán)隊(duì)的同學(xué)。以上便是我們?cè)谔詫氈饕谧龅氖虑椤? 前陣子,有些師弟師妹問我:在淘寶,前端開發(fā)主要是在做什么事情? 作為一個(gè)在淘寶已經(jīng)工作 5 年的「老兵」,我想我有資格來全面地回答一下這個(gè)問題,并通過這個(gè)機(jī)會(huì)向外部介紹一下我們團(tuán)隊(duì)...

    liuyix 評(píng)論0 收藏0
  • 也許你并不需要第三方小程序框架

    摘要:所以在小程序出現(xiàn)之后,一股框架之風(fēng)也很快的出現(xiàn),微信小程序剛推出之后,就出現(xiàn)了兩個(gè)比較出名的小程序開發(fā)框架,。 原文地址:https://ant-move.github.io/we... 這里說的去除小程序框架其實(shí)并不嚴(yán)謹(jǐn),因?yàn)樾〕绦虮旧硪菜闶且粋€(gè)框架,而且是一個(gè)功能更加完善的框架系統(tǒng)。在前端的概念中,我們一般說一個(gè)框架是指一個(gè)用來幫助開發(fā)者構(gòu)建用戶界面的框架,而小程序框架本身不僅僅包...

    red_bricks 評(píng)論0 收藏0
  • Chameleon跨端框架——壹個(gè)理想主義團(tuán)隊(duì)的開源作品

    摘要:跨端框架壹個(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...

    darkbug 評(píng)論0 收藏0
  • 前端每周清單半年盤點(diǎn)之 PWA 篇

    摘要:前端每周清單專注前端領(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ì)于...

    崔曉明 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<