欧美成人精品手机在线观看_69视频国产_动漫精品第一页_日韩中文字幕网 - 日本欧美一区二区

LOGO OA教程 ERP教程 模切知識交流 PMS教程 CRM教程 開發文檔 其他文檔  
 
網站管理員

HTMX:是在開歷史的倒車嗎?

admin
2023年11月18日 17:44 本文熱度 646

被 javascript 全面綁架的前端開發

十幾二十年前,我曾經是個自信滿滿的互聯網開發者。我可以輕松地使用 django 構建 Web UI。頁面上大大小小的重復部分,我都用 template 或者 fragment 抽象或者封裝。如果需要,我并不排斥撰寫 javascript 來增加交互性:

然而,這種方式構建的 UI 會導致用戶和頁面的每次交互都需要后端重新發送完整的 html 頁面,這既浪費帶寬,交互的方式又笨拙不流暢。因而,一些 ajax 庫便被創造出來提升交互能力。漸漸地,javascript 處理的事情越來越多,就連服務器端渲染 HTML template 的動作也慢慢遷移到了客戶端。最終,以 react 為代表的響應式組件化 UI 的春天來臨了:

react 帶給 web 開發很多革命性的理念:虛擬 dom,單向數據流,JSX 以及組件化思維。它讓前端從 HTML 客戶端徹底倒向了 Javascript 客戶端,同時讓后端退出前端渲染的舞臺,把生成 HTML 的主導權讓渡給前端,自己安安心心地只做數據 API 的提供方。但當 javascript 開始接管一切,HTML 不得不成為二等公民后,一切也隨之變了味 —— 連 footer 這樣完全由靜態 HTML 組成的內容都要通過 javascript (jsx) 完成。原本豐腴的 HTML 頁面瘦成一道閃電,body 里只剩下一個用來 mount 的根元素。

在 react 席卷前端世界之時,react 的缺陷便一個個暴露出來。于是,新的思想,新的框架,新的生態工具被創造出來,熱情的前端開發者架們以一種「逢山開路,遇水搭橋」的方式一路蒙眼狂奔,缺什么補什么:沒有合適的狀態管理,就創造出 redux;狀態管理太復雜,那引入 hooks;js 客戶端對 SEO 不友好,上 SSR …。這樣不斷堆疊解決方案后,最初簡潔明了的方案被硬生生折騰成一個龐雜的縫合怪。此時,我已經輕易不敢碰前端了,原本簡簡單單能搞定的事情,現在繁文縟節一大堆,寫點前端代碼我感覺自己都要被過度的復雜性壓得透不過起來。

更糟糕的是,由于 javascript 接管一切帶來的前端項目的大型化,使得 typescript 成為了最佳實踐。我并非貶低 typescript,事實上 typescript 是一門設計良好的語言,它很好地解決了 javascript 在大型前端項目中使用的諸多問題。但我們真的到處都需要「大型」前端項目么?導致前端項目如此龐雜臃腫的根源是什么?最初這些框架的主要目的難道不是為了讓前端更加響應式,更容易復用,更容易表達么?可如今,react 及那些前前后后崛起的前端框架們,包括 vue,solidjs,svelte 等等,都在以自身的復雜性迫使前端開發者,或者說像我這樣的「偽前端開發者」,不得不把小型項目大型化,簡單項目復雜化,于是應對復雜項目的 typescript 成為了必然的選擇。

被安在紀伯倫身上的一句中文名句:”我們已經走得太遠,以至于忘了為什么出發” 形象地描述了這十多年來前端的發展。我不烙卸嗌僨岸順絳蛟倍鄖岸說南腫錘械鉸猓裎藝庋惺焙蚪黿鍪竅胛約鶴齙南低程峁┮桓黽蚪嗟� UI —— 只需一茶匙就能裝下的前端需求 —— 卻面對 react 全家桶的復雜性產生深深的無力感。

差不多一年前,我在做一個后端低代碼的玩具項目時無意發現了 htmx,一下子就被其純凈的思想深深吸引。彼時我并未深入研究。過去兩周,因為工作的原因我迫切需要做點前端的工作時,我果斷地撿起了 htmx 進行深入試驗。兩周斷斷續續的開發過程中,我使用 axum (web server) + askama (template) + htmx + tailwindcss 很快地完成了我想做的事情,并且對界面高效地進行了好幾版迭代。我自己的感覺是:htmx 即便不能成長為前端的新勢力,它也能重塑所有非前端工程師對前端開發的信心。對于那些苦前端久矣的開發者來說,我們也許迎來了前端的 1984 時刻。

回歸 HTML 初心的 HTMX

雖然我找不到 HTMX 的名字的來源,根據它的愿景,我猜測它有 HTML eXtension 的意思。HTMX 認為我們應該增強和發展 HTML,HTML 的很多缺陷可以通過更好地 HTML 語義,比如標簽的屬性來彌補,而非直接讓 javascript 取代 HTML。這是 htmx.org 上的直接介紹:

htmx gives you access to AJAXCSS TransitionsWebSockets and Server Sent Events directly in HTML, using attributes, so you can build modern user interfaces with the simplicity and power of hypertext

HTMX 的核心愿景和特點包括:

  1. 逐步增強: HTMX 是基于逐步增強的原則設計的,這意味著它的目標是首先確保頁面在沒有 Javascript 的情況下可以工作,然后逐漸增加更多的功能。

  1. 低侵入性: HTMX 旨在盡可能少地侵入你的代碼。你不需要重寫整個應用來開始使用HTMX;相反,你可以只在需要的地方添加一些屬性。

  1. 純 HTML: 使用 HTMX,你可以在不編寫 Javascript 的情況下實現許多復雜的前端功能。這使得代碼更容易理解和維護,尤其是對于那些更喜歡或更熟悉 HTML 和服務器端編程的開發者。

  1. 與現有技術兼容: HTMX 可以與你已經使用的框架和庫無縫地協同工作。無論你是否使用 Flask、Django、Rails 或其他后端框架,HTMX 都可以簡單地嵌入其中。

  1. 小而快: 相比于許多現代的前端框架,HTMX 是非常輕量級的,這意味著它加載得更快,對用戶的體驗有所提升。

我們可以看到,HTMX 的目標是簡化前端開發,使開發者能夠快速、高效地創建交互性強、響應迅速的網頁,同時避免涉及大量的 Javascript 或復雜的前端框架。

當你不需要復雜的前端框架和大量的 javascript 開發時,你會發現,目前前端所面臨的很多問題都不是問題:不需要把整個頁面 javascript 化,不需要為了解決頁面 javascript 化引入的 SEO 問題,更不需要管理管理復雜的狀態,以及引入 typescript 來解決工程化的問題。

talk is cheat, show me the code!

我們先來看看 htmx 下,如何實現典型的前端功能:autocomplete。

不要過于震驚,這一小段代碼就是 HTMX 版 autocomplete 的全部代碼。可以看到,HTMX 給普通 HTML 標簽增加了幾個重要的屬性:

  • hx-trigger:用于指定何時以及如何觸發一個 htmx 動作,例如 AJAX 調用。通過這個屬性,開發者可以控制在某些事件發生時(例如,點擊、輸入或聚焦等),如何發起與服務器的交互。在這個例子里,我特意寫了兩個觸發事件,a) keyup 事件導致 input 變化時,延遲 500ms 后觸發;b) 接收到名為 custom-event 的事件。

  • hx-get:當 htmx 動作被觸發時,執行的調用。hx-get 代表 GET 請求,同理,你可以使用 hx-post,hx-put,hx-delete,hx-patch 等服務器調用。通過 hx-get 這樣的屬性,HTMX 把與服務器交互的權利下放給每一個標簽,而非傳統上那樣 —— 只有 <a/><form/> 才能和服務器交互。這可能是很多時候我們不得不引入 javascript 的一大原因。

  • hx-target:當服務器的響應返回時,響應被填充在哪個位置。hx-target 可以是任何 css 表達式,這里我們將其指向了 id 為 search-results 的節點。默認是當前節點。如果說 hx-get 這樣的屬性提供了頁面中無處不在與服務器交互的能力,那么 hx-target 就提供了頁面中無處不在的動態更新能力。這中動態更新能力是我們引入 javascript 的重要原因。

  • hx-swap:當服務器的相應返回時,內容該如何交換或者替換,默認是 innerHTML,也就是說 #search-results 內部的 HTML 會被服務器返回的數據替換。hx-swap 還有其他行為,比如 outerHTML,beforeend,afterend,甚至還可以添加如何做 swap 的動畫效果,大家可以自行查看文檔。

這幾個是對初學者而言最有用的屬性,掌握了它們就能處理大部分的頁內交互。HTMX 還提供了很多 hx-* 屬性,我就不一一介紹了。使用這些屬性,我們就可以控制搜索框的行為,很輕松地完成原本要不少 javascript 才能達到的效果。

我們再來看一個復雜一些的例子:

假設應用展示若干 note books,每個 note book 有若干 notes,每個 note 有詳盡的信息。我們用三欄式展示。用戶點擊最左欄的 book1 時,book1 下的 notes 以分頁的形式展示在第二欄,然后第二欄的第一個 note 的詳情在第三欄展示。

你可以想象一下這樣的頁面和交互需求用 react 該如何完成。

使用 HTMX,我們可以完全依照服務器渲染的思路設計,不必過多考慮客戶端如何維持狀態,如何動態刷新。

在第一次生成這個頁面的時候,我們可以把 book1 下面的所有 note summary 展示出來,然后再把 book1 note1 下面的 detail 也展示出來。幾個部分的模板片段如下。首先是左欄:

<ul>

{% for book in books %}

<li>

  <a hx-get="/books/{{book.id}}" hx-target="#note-list">{{book.name}}</a>

</li>

{% endfor %}

</ul>


然后中欄:

<div id="note-list">

{% for note in current_book.notes %}

<div onclick="htmx.trigger('#note-detail', 'loadNote', {id: '{{note.id}}'})">

  <h2>{{note.title}}</h2>

  <p>{{note.summary}}</p>

</div>

{% endfor %}

</div>


最后右欄:

<div id="note-detail"

      _="on loadNote(data) from body

      htmx.ajax('GET', `/notes/${data.id}`, '#note-detail')">

  <h3>{{title}}</h3>

  <p>{{detail}}</p>

</div>


這樣簡簡單單幾個模板,輔以額外的 HTMX 屬性,不光是第一次頁面渲染的結果有了,頁面也能根據用戶的點擊進行更新。比如用戶點擊 book2,它會觸發一個 GET 請求,訪問 /books/2,返回如下響應(就是中欄的模板生成的內容):

200 OK

HX-Trigger: {"loadNote": {"id": "book2id1"}}

Content-Type: text/html
<div onclick="htmx.trigger('#note-detail', 'loadNote', {id: 'book2id1'})">

  <h2>Hello 1</h2>

  <p>World 1</p>

</div>

<div onclick="htmx.trigger('#note-detail', 'loadNote', {id: 'book2id2'})">

  <h2>Hello 2</h2>

  <p>World 2</p>

</div>

...


這個結果會被 HTMX 渲染到 #note-list 中。于是中欄得到更新。

同時,因為返回的 HX-Trigger 頭帶了 loadNote 事件,該事件被 #node-detail 捕獲并發送 GET 請求到 /notes/book2id1 ,然后其響應被渲染到右欄。

一個事件導致頁面多處更新,這種并不簡單的處理,我們用 HTMX 輕松搞定了。

這里我們引入了一個新的東西:特殊的 HTTP 頭 HX-Trigger。HTMX 定義了很多新的 HTTP header,用于客戶端和服務器交互額外信息。這里的 HX-Trigger 頭,提供了一個強大而靈活的服務器端響應觸發客戶端事件的能力。

你可能會對這段代碼感到疑惑:

<div id="note-detail"

      _="on loadNote(data) from body

      htmx.ajax('GET', `/notes/${data.id}`, '#note-detail')"

>


它是 HTMX 的 hyperscript 的表述,等價于:

document.body.addEventListener("loadNote", function(e){

    htmx.ajax('GET', `/notes/${e.detail.id`, '#node-detail');

})


到目前為止,我們寫了兩三行非常簡單的 javascript,就實現了整個三欄加載和更新邏輯。我們用圖把整個邏輯梳理一下:

是不是相當簡潔?你是愿意撰寫這樣的代碼,還是原意從 npm init 開始,一步步設置 react 全家桶,最終寫上一大堆組件,維護一系列狀態,才能達成相同的目標?

對于上述這樣一個事件多處更新的場景,使用事件機制是我個人比較喜歡的實現。其實 HTMX 也提供了其他解決方案,比如使用 hx-swap-oob,或者 multi-swap 擴展,寫起來比我上述的方案還要簡潔一些。

回顧上述兩個例子,我們可以看到,在使用 HTMX 后,大量的邏輯依舊保留在后端,就像十幾年前我們在 rails/django 里處理的那樣。我們把一個個 template / fragment 拆分到組件級別,然后把服務器渲染好的 HTML 傳遞給客戶端。只不過,有了 HTMX 后,我們可以很輕松地實現響應式前端,所有的操作都可以以你需要的粒度更新在頁面的任何位置。

由于 HTMX 用標簽屬性這樣一種很舒服的方式來標準化基本的客戶端/服務器間的操作,在大多數場合下,配合 tailwindcss 這樣的 CCS 工具箱,構建前端只需要和 HTML 打交道。在我做項目時,我基本上就是找 flowbite 這樣的網站上的某個組件的示例代碼,稍作修改使其模板化,再把這些模板整合起來,一個個頁面就構建出來了。我再也不需要拘泥于究竟要做 SPA 還是 MPA,一切根據需求隨心而動。

當然,使用 HTMX 也可能會帶來一些耦合性問題 —— 這并非 HTMX 的鍋,而是自 PHP 起,所有做服務端渲染 HTML 的后端都會帶來的問題:邏輯層和表現層的耦合,以及多端的支持。

邏輯層和表現層的耦合可以通過更好地架構設計(或者引入合適的框架)來避免,我們放下不表。多端的支持可以通過服務器對內容協商的支持而得到支持。比如 web 端,可以發送 Accept: text/htmx, text/html,明確要求 HTMX 格式或者 HTML 格式的數據,而移動端,可以發送 Accept: application/json 來獲得 JSON 數據。具體服務端的實現邏輯(axum版)如下:

總結

HTMX 為非前端工程師重新打開了前端開發的大門。如果你不是開發像 spreadsheet,google map 這樣的重交互應用,基本上,你都能很好地用 HTMX 來取代現有的前端開發框架,重新回到以 HTML 為中心的輕量級前端開發上。你不必拘泥于客戶端究竟該實現成 SPA 還是 MPA,可以用最合適的方式路由,最自然的方式展示數據,讓用戶跟數據交互(無論是增刪改查還是其他什么動作)。

目前,HTMX 的生態還剛剛起步,我非常期待主流的后端框架對其進行深度的支持甚至整合。我相信隨著 HTMX 的價值被不斷發掘出來,最終,非前端開發者可以重拾信心,無痛開發一個包含 web 前端的完整的產品。


該文章在 2023/11/18 17:47:23 編輯過
關鍵字查詢
相關文章
正在查詢...
點晴ERP是一款針對中小制造業的專業生產管理軟件系統,系統成熟度和易用性得到了國內大量中小企業的青睞。
點晴PMS碼頭管理系統主要針對港口碼頭集裝箱與散貨日常運作、調度、堆場、車隊、財務費用、相關報表等業務管理,結合碼頭的業務特點,圍繞調度、堆場作業而開發的。集技術的先進性、管理的有效性于一體,是物流碼頭及其他港口類企業的高效ERP管理信息系統。
點晴WMS倉儲管理系統提供了貨物產品管理,銷售管理,采購管理,倉儲管理,倉庫管理,保質期管理,貨位管理,庫位管理,生產管理,WMS管理系統,標簽打印,條形碼,二維碼管理,批號管理軟件。
點晴免費OA是一款軟件和通用服務都免費,不限功能、不限時間、不限用戶的免費OA協同辦公管理系統。
Copyright 2010-2025 ClickSun All Rights Reserved