<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          使用 Node.js 和 htmx 構(gòu)建全棧 CRUD 應(yīng)用程序

          共 24359字,需瀏覽 49分鐘

           ·

          2024-04-11 22:10

          7d81195c60313da505afbc87951c639c.webp

          在本教程中,我將演示如何使用 Node 作為后端和 htmx 作為前端來構(gòu)建功能齊全的 CRUD 應(yīng)用程序。這將演示 htmx 如何集成到全棧應(yīng)用程序中,使您能夠評估其有效性并確定它是否是您未來項(xiàng)目的不錯選擇。

          htmx 是一個現(xiàn)代 JavaScript 庫,旨在通過實(shí)現(xiàn)部分 HTML 更新來增強(qiáng) Web 應(yīng)用,而無需重新加載整個頁面。與傳統(tǒng)前端框架中的 JSON 有效載荷不同,它通過有線方式發(fā)送 HTML 來實(shí)現(xiàn)這一功能。

          我們將要構(gòu)建什么

          我們將開發(fā)一個簡單的聯(lián)系人管理器,能夠執(zhí)行所有 CRUD 操作:創(chuàng)建、讀取、更新和刪除聯(lián)系人。通過利用 htmx,該應(yīng)用程序?qū)⑻峁﹩雾搼?yīng)用程序 (SPA) 的感覺,從而增強(qiáng)交互性和用戶體驗(yàn)。

          如果用戶禁用 JavaScript,應(yīng)用程序?qū)⒁哉撍⑿碌姆绞竭\(yùn)行,從而保持可用性和可發(fā)現(xiàn)性。這種方法展示了 htmx 創(chuàng)建現(xiàn)代 Web 應(yīng)用程序的能力,同時保持它們的可訪問性和 SEO 友好性。

          這就是我們最終得到的結(jié)果。

          dd300c2d4205d8d698cd177d7eb63e6a.webp

          本文的代碼可以在隨附的 GitHub 存儲庫[1]中找到。

          先決條件

          要學(xué)習(xí)本教程,您需要在 PC 上安裝 Node.js。如果您尚未安裝 Node,請前往官方 Node 下載頁面并獲取適合您系統(tǒng)的正確二進(jìn)制文件。或者,您可能想使用版本管理器安裝 Node。這種方法允許您安裝多個 Node 版本并在它們之間隨意切換。

          除此之外,熟悉 Node、Pug(我們將使用它們作為模板引擎)和 htmx 會有所幫助,但不是必需的。如果您想復(fù)習(xí)以上任何內(nèi)容,請查看我們的教程:使用 Node 構(gòu)建簡單的初學(xué)者應(yīng)用程序[2]、Pug HTML 模板預(yù)處理器指南[3]和 htmx 簡介[4]。

          在開始之前,請運(yùn)行以下命令:

                
                node -v
          npm -v

          您應(yīng)該看到如下輸出:

                
                v20.11.1
          10.4.0

          這確認(rèn)了 Node 和 npm 已安裝在您的計算機(jī)上,并且可以從命令行環(huán)境進(jìn)行訪問。

          設(shè)置項(xiàng)目

          讓我們從搭建一個新的 Node 項(xiàng)目開始:

                
                mkdir contact-manager
          cd contact-manager
          npm init -y

          這應(yīng)該在項(xiàng)目根目錄中創(chuàng)建一個 package.json 文件。

          接下來,讓我們安裝我們需要的依賴項(xiàng):

                
                npm i express method-override pug

          在這些包中,Express 是我們應(yīng)用程序的支柱。它是一個快速且簡約的 Web 框架,提供了一種簡單的方法來處理請求和響應(yīng),并將 URL 路由到特定的處理函數(shù)。 Pug 將充當(dāng)我們的模板引擎,而我們將使用方法覆蓋在客戶端不支持的地方使用 HTTP 動詞,例如 PUT 和 DELETE。

          接下來,在根目錄中創(chuàng)建一個 app.js 文件:

                
                touch app.js

          并添加以下內(nèi)容:

                
                const express = require('express');
          const path = require('path');
          const routes = require('./routes/index');

          const app = express();

          app.set('views', path.join(__dirname, 'views'));
          app.set('view engine''pug');

          app.use(express.static('public'));
          app.use('/', routes);

          const server = app.listen(3000, () => {
            console.log(`Express is running on port ${server.address().port}`);
          });

          在這里,我們正在設(shè)置 Express 應(yīng)用程序的結(jié)構(gòu)。這包括將 Pug 配置為渲染視圖的視圖引擎、定義靜態(tài)資產(chǎn)的目錄以及連接路由器。

          該應(yīng)用程序偵聽端口 3000,并使用控制臺日志來確認(rèn) Express 正在運(yùn)行并準(zhǔn)備好處理指定端口上的請求。此設(shè)置構(gòu)成了我們應(yīng)用程序的基礎(chǔ),并準(zhǔn)備好通過更多功能和路由進(jìn)行擴(kuò)展。

          接下來,讓我們創(chuàng)建路由文件:

                
                mkdir routes
          touch routes/index.js

          打開該文件并添加以下內(nèi)容:

                
                const express = require('express');
          const router = express.Router();

          // GET /contacts
          router.get('/contacts'async (req, res) => {
            res.send('It works!');
          });

          在這里,我們在新創(chuàng)建的路由目錄中設(shè)置基本路由。此路由在 /contacts 端點(diǎn)偵聽 GET 請求,并使用簡單的確認(rèn)消息進(jìn)行響應(yīng),表明一切正常。

          接下來,使用以下內(nèi)容更新 package.json 文件的“scripts”部分:

                
                "scripts": {
            "dev""node --watch app.js"
          },

          這利用了 Node.js 中的新監(jiān)視模式,只要檢測到任何更改,該模式就會重新啟動我們的應(yīng)用程序。

          最后,使用 npm run dev 啟動所有內(nèi)容,然后在瀏覽器中訪問 http://localhost:3000/contacts/。您應(yīng)該會看到一條消息“It works!”。

          31fe546eda28922b9dbdec469b95a42f.webp

          激動人心的時刻!

          顯示所有聯(lián)系人

          現(xiàn)在讓我們添加一些要顯示的聯(lián)系人。由于我們專注于 htmx,因此為了簡單起見,我們將使用硬編碼數(shù)組。這將使事情變得精簡,使我們能夠?qū)W⒂?htmx 的動態(tài)功能,而無需復(fù)雜的數(shù)據(jù)庫集成。

          對于那些有興趣稍后添加數(shù)據(jù)庫的人來說,SQLite 和 Sequelize 是不錯的選擇,它們提供了不需要單獨(dú)數(shù)據(jù)庫服務(wù)器的基于文件的系統(tǒng)。

          話雖如此,請將以下內(nèi)容添加到第一個路由之前的 index.js 中:

                
                const contacts = [
            { id1name'John Doe'email'[email protected]' },
            { id2name'Jane Smith'email'[email protected]' },
            { id3name'Emily Johnson'email'[email protected]' },
            { id4name'Aarav Patel'email'[email protected]' },
            { id5name'Liu Wei'email'[email protected]' },
            { id6name'Fatima Zahra'email'[email protected]' },
            { id7name'Carlos Hernández'email'[email protected]' },
            { id8name'Olivia Kim'email'[email protected]' },
            { id9name'Kwame Nkrumah'email'[email protected]' },
            { id10name'Chen Yu'email'[email protected]' },
          ];

          現(xiàn)在,我們需要為路由創(chuàng)建一個顯示模板。創(chuàng)建一個包含 index.pug 文件的 views 文件夾:

                
                mkdir views
          touch views/index.pug

          并添加以下內(nèi)容:

                
                doctype html
          html
          head
          meta(charset='UTF-8')
          title Contact Manager

          link(rel='preconnect', )
          link(rel='preconnect', , crossorigin)
          link(, rel='stylesheet')

          link(rel='stylesheet', href='/styles.css')
          body
          header
          a(href='/contacts')
          h1 Contact Manager

          section#sidebar
          ul.contact-list
          each contact in contacts
          li #{contact.name}
          div.actions
          a(href='/contacts/new') New Contact

          main#content
          p Select a contact

          script(src='https://unpkg.com/[email protected]')

          在此模板中,我們?yōu)閼?yīng)用程序布置 HTML 結(jié)構(gòu)。在 head 部分,我們包含了來自 Google Fonts 的 Roboto 字體和自定義樣式的樣式表。

          正文分為標(biāo)題、用于列出聯(lián)系人的側(cè)邊欄以及用于存放所有聯(lián)系信息的主要內(nèi)容區(qū)域。內(nèi)容區(qū)域當(dāng)前包含一個占位符。在正文的末尾,我們還包含來自 CDN 的最新版本的 htmx 庫。

          該模板期望接收一個聯(lián)系人數(shù)組(在 contacts 變量中),我們在側(cè)邊欄中對其進(jìn)行迭代,并使用 Pug 的插值語法在無序列表中輸出每個聯(lián)系人姓名。

          接下來,讓我們創(chuàng)建自定義樣式表:

                
                mkdir public
          touch public/styles.css

          我不想在這里列出樣式。請從隨附的 GitHub 存儲庫中的 CSS 文件[5]中復(fù)制它們,或者隨意添加一些您自己的 CSS 文件。 ??

          回到 index.js,更新路由以使用模板:

                
                // GET /contacts
          router.get('/contacts', (req, res) => {
            res.render('index', { contacts });
          });

          現(xiàn)在,當(dāng)您刷新頁面時,您應(yīng)該會看到類似這樣的內(nèi)容。

          f64e20eb2980dbdb506f84d07a2d42b4.webp

          顯示單個聯(lián)系人

          到目前為止,我們所做的只是建立了一個基本的 Express 應(yīng)用程序。讓我們改變一下,最后添加 htmx。下一步要做的是,當(dāng)用戶點(diǎn)擊側(cè)邊欄中的聯(lián)系人時,該聯(lián)系人的信息就會顯示在主內(nèi)容區(qū)域--自然不需要重新載入整個頁面。

          首先,讓我們將側(cè)邊欄移至其自己的模板中:

                
                touch views/sidebar.pug

          將以下內(nèi)容添加到這個新文件中:

                
                ul.contact-list
          each contact in contacts
          li
          a(
          href=`/contacts/${contact.id}`,
          hx-get=`/contacts/${contact.id}`,
          hx-target='#content',
          hx-push-url='true'
          )= contact.name

          div.actions
          a(href='/contacts/new') New Contact

          這里我們?yōu)槊總€聯(lián)系人創(chuàng)建了一個指向 /contacts/${contact.id} 的鏈接,并添加了三個 htmx 屬性:

          • hx-get:當(dāng)用戶單擊鏈接時,htmx 將攔截單擊并通過 Ajax 向 /contacts/${contact.id} 端點(diǎn)發(fā)出 GET 請求。
          • hx-target:當(dāng)請求完成時,響應(yīng)將被插入到 ID 為 content 的 div 中。我們在這里沒有指定任何類型的交換策略,因此 div 的內(nèi)容將被 Ajax 請求返回的內(nèi)容替換。這是默認(rèn)行為。
          • hx-push-url:這將確保 htx-get 中指定的值被推送到瀏覽器的歷史堆棧中,從而更改 URL。

          更新 index.pug 以使用我們的模板:

                
                section#sidebar
          include sidebar.pug

          請記住:Pug 對空格敏感,因此請務(wù)必使用正確的縮進(jìn)。

          現(xiàn)在讓我們在 index.js 中創(chuàng)建一個新端點(diǎn)以返回 htmx 期望的 HTML 響應(yīng):

                
                // GET /contacts/1
          router.get('/contacts/:id', (req, res) => {
            const { id } = req.params;
            const contact = contacts.find((c) => c.id === Number(id));

            res.send(`
              <h2>${contact.name}</h2>
              <p><strong>Name:</strong> ${contact.name}</p>
              <p><strong>Email:</strong> ${contact.email}</p>
            `
          );
          });

          如果保存并刷新瀏覽器,您現(xiàn)在應(yīng)該能夠查看每個聯(lián)系人的詳細(xì)信息。

          dd300c2d4205d8d698cd177d7eb63e6a.webp

          網(wǎng)絡(luò)上的 HTML

          讓我們花點(diǎn)時間了解一下這里發(fā)生了什么。正如文章開頭提到的,htmx 通過網(wǎng)絡(luò)傳輸 HTML,而不是傳統(tǒng)前端框架的 JSON 有效負(fù)載。

          如果我們打開瀏覽器的開發(fā)人員工具,切換到“Network”選項(xiàng)卡并單擊其中一個聯(lián)系人,我們就可以看到這一點(diǎn)。收到來自前端的請求后,我們的 Express 應(yīng)用程序會生成顯示該聯(lián)系人所需的 HTML,并將其發(fā)送到瀏覽器,其中 htmx 將其交換到 UI 中的正確位置。

          29e7ca30440882fa69822ccbbc522574.webp

          處理全頁刷新

          所以事情進(jìn)展得很順利,是吧?感謝 htmx,我們通過在錨標(biāo)記上指定幾個屬性來使頁面動態(tài)化。不幸的是,有一個問題……

          如果您顯示聯(lián)系人,然后刷新頁面,我們可愛的用戶界面就會消失,您看到的只是裸露的聯(lián)系方式詳細(xì)信息。如果您直接在瀏覽器中加載 URL,也會發(fā)生同樣的情況。

          如果你仔細(xì)想想,其原因是顯而易見的。當(dāng)您訪問 http://localhost:3000/contacts/1 之類的 URL 時, '/contacts/:id' 的 Express 路由將啟動并返回聯(lián)系人的 HTML,正如我們告訴它的那樣。它對我們用戶界面的其余部分一無所知。

          為了解決這個問題,我們需要做一些改動。在服務(wù)器上,我們需要檢查是否存在 HX-Request 標(biāo)頭,它表明請求來自 htmx。如果存在,我們就可以發(fā)送部分內(nèi)容。否則,我們需要發(fā)送整個頁面。

          像這樣更改路由處理程序:

                
                // GET /contacts/1
          router.get('/contacts/:id', (req, res) => {
            const { id } = req.params;
            const contact = contacts.find((c) => c.id === Number(id));

            if (req.headers['hx-request']) {
              res.send(`
                <h2>${contact.name}</h2>
                <p><strong>Name:</strong> ${contact.name}</p>
                <p><strong>Email:</strong> ${contact.email}</p>
              `
          );
            } else {
              res.render('index', { contacts });
            }
          });

          現(xiàn)在,當(dāng)您重新加載頁面時,用戶界面不會消失。但是,它確實(shí)會從您正在查看的任何聯(lián)系人恢復(fù)為消息“選擇聯(lián)系人”,這并不理想。

          為了解決這個問題,我們可以在 index.pug 模板中引入 case 語句:

                
                main#content
          case action
          when 'show'
          h2 #{contact.name}
          p #[strong Name:] #{contact.name}
          p #[strong Email:] #{contact.email}
          when 'new'
          // Coming soon
          when 'edit'
          // Coming soon
          default
          p Select a contact

          最后更新路由處理程序:

                
                if (req.headers['hx-request']) {
            // As before
          else {
            res.render('index', { action'show', contacts, contact });
          }

          請注意,我們現(xiàn)在傳入一個 contact 變量,該變量將在整個頁面重新加載時使用。

          這樣,我們的應(yīng)用程序應(yīng)該能夠承受刷新或直接加載聯(lián)系人。

          快速重構(gòu)

          雖然這樣做可行,但您可能會注意到,我們的路由處理程序和主 pug 模板中都有一些重復(fù)的內(nèi)容。這種情況并不理想,只要聯(lián)系人的屬性不超過幾個,或者我們需要使用一些邏輯來決定顯示哪些屬性,事情就會開始變得臃腫。

          為了解決這個問題,讓我們將聯(lián)系人移動到自己的模板中:

                
                touch views/contact.pug

          在新創(chuàng)建的模板中,添加以下內(nèi)容:

                
                h2 #{contact.name}

          p #[strong Name:] #{contact.name}
          p #[strong Email:] #{contact.email}

          在主模板( index.pug )中:

                
                main#content
          case action
          when 'show'
          include contact.pug

          還有我們的路由處理程序:

                
                if (req.headers['hx-request']) {
            res.render('contact', { contact });
          else {
            res.render('index', { action'show', contacts, contact });
          }

          事情應(yīng)該仍然像以前一樣工作,但現(xiàn)在我們已經(jīng)刪除了重復(fù)的代碼。

          新的聯(lián)系表

          我們要關(guān)注的下一個任務(wù)是創(chuàng)建新聯(lián)系人。本教程的這一部分將指導(dǎo)您設(shè)置表單和后端邏輯,使用 htmx 動態(tài)處理提交。

          讓我們從更新側(cè)邊欄模板開始。更改:

                
                div.actions
          a(href='/contacts/new') New Contact

          … 到:

                
                div.actions
          a(
          href='/contacts/new',
          hx-get='/contacts/new',
          hx-target='#content',
          hx-push-url='true'
          ) New Contact

          這將使用與鏈接相同的 htmx 屬性來顯示聯(lián)系人:hx-get 將通過 Ajax 向 /contacts/new 端點(diǎn)發(fā)出 GET 請求,hx-target 將指定插入響應(yīng)的位置,hx-push-url 將確保更改 URL。

          現(xiàn)在讓我們?yōu)楸韱蝿?chuàng)建一個新模板:

                
                touch views/form.pug

          并添加以下代碼:

                
                h2 New Contact

          form(
          action='/contacts',
          method='POST',
          hx-post='/contacts',
          hx-target='#sidebar',
          hx-on::after-request='if(event.detail.successful) this.reset()'
          )
          label(for='name') Name:
          input#name(type='text', name='name', required)

          label(for='email') Email:
          input#email(type='email', name='email', required)

          div.actions
          button(type='submit') Submit

          在這里,我們使用 hx-post 屬性告訴 htmx 攔截表單提交,并向 /contacts 端點(diǎn)發(fā)出帶有表單數(shù)據(jù)的 POST 請求。結(jié)果(更新的聯(lián)系人列表)將被插入到側(cè)邊欄中。在這種情況下,我們不想更改 URL,因?yàn)橛脩艨赡芟胍斎攵鄠€新聯(lián)系人。但是,我們確實(shí)希望在成功提交后清空表單,這就是 hx-on::after-request 的作用。 hx-on* 屬性允許您內(nèi)聯(lián)嵌入腳本以直接響應(yīng)元素上的事件。你可以在這里讀更多[6]關(guān)于它的內(nèi)容。

          接下來,我們在 index.js 中添加表單的路由:

                
                // GET /contacts
          ...

          // GET /contacts/new
          router.get('/contacts/new', (req, res) => {
            if (req.headers['hx-request']) {
              res.render('form');
            } else {
              res.render('index', { action'new', contacts, contact: {} });
            }
          });

          // GET /contacts/1
          ...

          路由順序在這里很重要。如果您先有 '/contacts/:id' 路由,那么 Express 將嘗試查找 ID 為 new 的聯(lián)系人。

          最后,更新我們的 index.pug 模板以使用以下形式:

                
                when 'new'
          include form.pug

          刷新頁面,此時您應(yīng)該能夠通過單擊側(cè)欄中的“New Contact”鏈接來呈現(xiàn)新的聯(lián)系人表單。

          7064cb4c7da5c507702df8dc84613504.webp

          創(chuàng)建聯(lián)系人

          現(xiàn)在我們需要創(chuàng)建一個路由來處理表單提交。

          首先更新 app.js 以使我們能夠訪問路由處理程序中的表單數(shù)據(jù)。

                
                const express = require('express');
          const path = require('path');
          const routes = require('./routes/index');

          const app = express();

          app.set('views', path.join(__dirname, 'views'));
          app.set('view engine''pug');

          + app.use(express.urlencoded({ extendedtrue }));
          app.use(express.static('public'));
          app.use('/', routes);

          const server = app.listen(3000, () => {
            console.log(`Express is running on port ${server.address().port}`);
          });

          以前,我們會使用 body-parser 包,但我最近了解到這不再是必要的。

          然后將以下內(nèi)容添加到 index.js

                
                // POST /contacts
          router.post('/contacts', (req, res) => {
            const newContact = {
              id: contacts.length + 1,
              name: req.body.name,
              email: req.body.email,
            };

            contacts.push(newContact);

            if (req.headers['hx-request']) {
              res.render('sidebar', { contacts });
            } else {
              res.render('index', { action'new', contacts, contact: {} });
            }
          });

          在這里,我們使用從客戶端收到的數(shù)據(jù)創(chuàng)建一個新聯(lián)系人,并將其添加到 contacts 數(shù)組中。然后,我們重新渲染側(cè)邊欄,并向其傳遞更新的聯(lián)系人列表。

          請注意,如果您正在制作任何類型的有用戶的應(yīng)用程序,則由您負(fù)責(zé)驗(yàn)證從客戶端接收的數(shù)據(jù)。在我們的示例中,我添加了一些基本的客戶端驗(yàn)證,但這很容易被繞過。

          我上面鏈接的 Node 教程中有一個示例,說明如何使用 express-validator[7] 包來驗(yàn)證服務(wù)器上的輸入。

          現(xiàn)在,如果您刷新瀏覽器并嘗試添加聯(lián)系人,它應(yīng)該按預(yù)期工作:新聯(lián)系人應(yīng)添加到側(cè)邊欄,并且應(yīng)重置表單。

          添加 toast 消息提示

          這很好,但現(xiàn)在我們需要一種方法來通知用戶聯(lián)系人已添加。在典型的應(yīng)用程序中,我們會使用 toast 消息——一種臨時通知,提醒用戶操作的結(jié)果。

          我們使用 htmx 遇到的問題是,我們在成功創(chuàng)建新聯(lián)系人后更新側(cè)邊欄,但這不是我們希望顯示 toast 消息的位置。更好的位置將位于新聯(lián)系表格上方。

          為了解決這個問題,我們可以使用 hx-swap-oob 屬性。這允許您指定響應(yīng)中的某些內(nèi)容應(yīng)交換到目標(biāo)以外的 DOM 中,即“Out of Band”。

          更新路由處理程序如下:

                
                if (req.headers['hx-request']) {
            res.render('sidebar', { contacts }, (err, sidebarHtml) => {
              const html = `
                <main id="content" hx-swap-oob="afterbegin">
                  <p class="flash">Contact was successfully added!</p>
                </main>
                ${sidebarHtml}
              `
          ;
              res.send(html);
            });
          else {
            res.render('index', { action'new', contacts, contact: {} });
          }

          在這里,我們像以前一樣渲染側(cè)邊欄,但向 render 方法傳遞一個匿名函數(shù)作為第三個參數(shù)。該函數(shù)接收通過調(diào)用 res.render('sidebar', { contacts }) 生成的 HTML,然后我們可以使用它來組裝最終響應(yīng)。

          通過指定交換策略 "afterbegin" ,將 toast 消息插入到容器的頂部。

          現(xiàn)在,當(dāng)我們添加聯(lián)系人時,我們應(yīng)該會收到一條不錯的消息,告訴我們發(fā)生了什么。

          de13a34eedd7c71108b76aad4322edf7.webp

          編輯聯(lián)系人

          為了更新聯(lián)系人,我們將重用上一節(jié)中創(chuàng)建的表單。

          讓我們首先更新 contact.pug 模板以添加以下內(nèi)容:

                
                div.actions
          a(
          href=`/contacts/${contact.id}/edit`,
          hx-get=`/contacts/${contact.id}/edit`,
          hx-target='#content',
          hx-push-url='true'
          ) Edit Contact

          這將在聯(lián)系人詳細(xì)信息下方添加一個編輯聯(lián)系人按鈕。正如我們之前所見,當(dāng)單擊鏈接時, hx-get 將通過 Ajax 向 /${contact.id}/edit 端點(diǎn)發(fā)出 GET 請求, hx-target 將指定插入位置響應(yīng), hx-push-url 將確保 URL 發(fā)生更改。

          現(xiàn)在讓我們更改 index.pug 模板以使用以下形式:

                
                when 'edit'
          include form.pug

          還添加一個路由處理程序來顯示表單:

                
                // GET /contacts/1/edit
          router.get('/contacts/:id/edit', (req, res) => {
            const { id } = req.params;
            const contact = contacts.find((c) => c.id === Number(id));

            if (req.headers['hx-request']) {
              res.render('form', { contact });
            } else {
              res.render('index', { action'edit', contacts, contact });
            }
          });

          請注意,我們使用請求中的 ID 檢索聯(lián)系人,然后將該聯(lián)系人傳遞到表單。

          我們還需要更新新的聯(lián)系人處理程序以執(zhí)行相同的操作,但此處傳遞一個空對象:

                
                // GET /contacts/new
          router.get('/contacts/new', (req, res) => {
            if (req.headers['hx-request']) {
          -    res.render('form');
          +    res.render('form', { contact: {} });
            } else {
              res.render('index', { action'new', contacts, contact: {} });
            }
          });

          然后我們需要更新表單本身:

                
                - isEditing = () => !(Object.keys(contact).length === 0);

          h2=isEditing() ? "Edit Contact" : "New Contact"

          form(
          action=isEditing() ? `/update/${contact.id}?_method=PUT` : '/contacts',
          method='POST',

          hx-post=isEditing() ? false : '/contacts',
          hx-put=isEditing() ? `/update/${contact.id}` : false,
          hx-target='#sidebar',
          hx-push-url=isEditing() ? `/contacts/${contact.id}` : false
          hx-on::after-request='if(event.detail.successful) this.reset()',
          )
          label(for='name') Name:
          input#name(type='text', name='name', required, value=contact.name)

          label(for='email') Email:
          input#email(type='email', name='email', required, value=contact.email)

          div.actions
          button(type='submit') Submit

          當(dāng)我們向此表單傳遞聯(lián)系人或空對象時,我們現(xiàn)在有一種簡單的方法來確定我們是否處于“編輯”或“創(chuàng)建”模式。我們可以通過檢查 Object.keys(contact).length 來做到這一點(diǎn)。我們還可以使用 Pug 的無緩沖代碼語法將此檢查提取到文件頂部的一個小輔助函數(shù)中。

          一旦我們知道自己所處的模式,我們就可以有條件地更改頁面標(biāo)題,然后決定向表單標(biāo)記添加哪些屬性。對于編輯表單,我們需要添加 hx-put 屬性并將其設(shè)置為 /update/${contact.id} 。保存聯(lián)系人詳細(xì)信息后,我們還需要更新 URL。

          為了做到這一切,我們可以利用這樣一個事實(shí):如果條件返回 false ,Pug 將從標(biāo)簽中省略該屬性。

          這意味著:

                
                form(
          action=isEditing() ? `/update/${contact.id}?_method=PUT` : '/contacts',
          method='POST',

          hx-post=isEditing() ? false : '/contacts',
          hx-put=isEditing() ? `/update/${contact.id}` : false,
          hx-target='#sidebar',
          hx-on::after-request='if(event.detail.successful) this.reset()',
          hx-push-url=isEditing() ? `/contacts/${contact.id}` : false
          )

          當(dāng) isEditing() 返回 false 時將編譯為以下內(nèi)容:

                
                <form
            action="/contacts"
            method="POST"
            hx-post="/contacts"
            hx-target="#sidebar"
            hx-on::after-request="if(event.detail.successful) this.reset()"
          >

            ...
          </form>

          但是當(dāng) isEditing() 返回 true 時,它將編譯為:

                
                <form
            action="/update/1?_method=PUT"
            method="POST"
            hx-put="/update/1"
            hx-target="#sidebar"
            hx-on::after-request="if(event.detail.successful) this.reset()"
            hx-push-url="/contacts/1"
          >

            ...
          </form>

          在其更新狀態(tài)下,請注意表單操作是 "/update/1?_method=PUT" 。添加此查詢字符串參數(shù)是因?yàn)槲覀冋谑褂梅椒ǜ采w包,它將使我們的路由器響應(yīng) PUT 請求。

          開箱即用的 htmx 可以發(fā)送 PUT 和 DELETE 請求,但瀏覽器卻不行。這意味著,如果我們要處理 JavaScript 被禁用的情況,就需要復(fù)制我們的路由處理程序,讓它同時響應(yīng) PUT(htmx)和 POST(瀏覽器)。使用這種中間件將使我們的代碼保持 DRY。

          讓我們繼續(xù)將其添加到 app.js

                
                const express = require('express');
          const path = require('path');
          const methodOverride = require('method-override');
          const routes = require('./routes/index');

          const app = express();

          app.set('views', path.join(__dirname, 'views'));
          app.set('view engine''pug');

          + app.use(methodOverride('_method'));
          app.use(express.urlencoded({ extendedtrue }));
          app.use(express.static('public'));
          app.use('/', routes);

          const server = app.listen(3000, () => {
            console.log(`Express is running on port ${server.address().port}`);
          });

          最后,讓我們用新的路由處理程序更新 index.js

                
                // PUT /contacts/1
          router.put('/update/:id', (req, res) => {
            const { id } = req.params;

            const newContact = {
              idNumber(id),
              name: req.body.name,
              email: req.body.email,
            };

            const index = contacts.findIndex((c) => c.id === Number(id));

            if (index !== -1) contacts[index] = newContact;

            if (req.headers['hx-request']) {
              res.render('sidebar', { contacts }, (err, sidebarHtml) => {
                res.render('contact', { contact: contacts[index] }, (err, contactHTML) => {
                  const html = `
                    ${sidebarHtml}
                    <main id="content" hx-swap-oob="true">
                      <p class="flash">Contact was successfully updated!</p>
                      ${contactHTML}
                    </main>
                  `
          ;

                  res.send(html);
                });
              });
            } else {
              res.redirect(`/contacts/${index + 1}`);
            }
          });

          希望現(xiàn)在沒有什么太神秘的事情了。在處理程序的開頭,我們從請求參數(shù)中獲取聯(lián)系人 ID。然后,我們找到想要更新的聯(lián)系人,并將其替換為根據(jù)我們收到的表單數(shù)據(jù)創(chuàng)建的新聯(lián)系人。

          在處理 htmx 請求時,我們首先用更新的聯(lián)系人列表呈現(xiàn)側(cè)邊欄模板。然后,我們用更新的聯(lián)系人渲染聯(lián)系人模板,并使用這兩次調(diào)用的結(jié)果來組合我們的響應(yīng)。與之前一樣,我們使用 “Out of Band” 更新創(chuàng)建一條 toast 消息,告知用戶聯(lián)系人已更新。

          此時,您應(yīng)該能夠更新聯(lián)系人。

          e4f98fb64834b8c0dc2532ce07adb988.webp

          刪除聯(lián)系人

          最后一個難題是刪除聯(lián)系人的能力。讓我們在聯(lián)系人模板中添加一個按鈕來執(zhí)行此操作:

                
                div.actions
          form(method='POST', action=`/delete/${contact.id}?_method=DELETE`)
          button(
          type='submit',
          hx-delete=`/delete/${contact.id}`,
          hx-target='#sidebar',
          hx-push-url='/contacts'
          class='link'
          ) Delete Contact

          a(
          // as before
          )

          請注意,最好使用表單和按鈕來發(fā)出 DELETE 請求。表單是為導(dǎo)致更改(例如刪除)的操作而設(shè)計的,這確保了語義的正確性。此外,使用鏈接進(jìn)行刪除操作可能存在風(fēng)險,因?yàn)樗阉饕婵赡軙o意中跟蹤鏈接,從而可能導(dǎo)致不必要的刪除。

          話雖這么說,我添加了一些 CSS 來將按鈕設(shè)置為鏈接樣式,因?yàn)榘粹o很難看。如果您之前從存儲庫復(fù)制了樣式,那么您的代碼中已經(jīng)包含了該樣式。

          最后,我們的路由處理程序在 index.js 中:

                
                // DELETE /contacts/1
          router.delete('/delete/:id', (req, res) => {
            const { id } = req.params;
            const index = contacts.findIndex((c) => c.id === Number(id));

            if (index !== -1) contacts.splice(index, 1);
            if (req.headers['hx-request']) {
              res.render('sidebar', { contacts }, (err, sidebarHtml) => {
                const html = `
                  <main id="content" hx-swap-oob="true">
                    <p class="flash">Contact was successfully deleted!</p>
                  </main>
                  ${sidebarHtml}
                `
          ;
                res.send(html);
              });
            } else {
              res.redirect('/contacts');
            }
          });

          刪除聯(lián)系人后,我們將更新側(cè)邊欄并向用戶顯示一條提示消息。

          21f81cae3dc955e7865aa9dc6a5d6e09.webp

          更進(jìn)一步

          就這樣吧。

          在本文中,我們使用 Node 和 Express 作為后端,使用 htmx 作為前端,制作了一個全棧 CRUD 應(yīng)用程序。在此過程中,我演示了 htmx 如何簡化向 Web 應(yīng)用程序添加動態(tài)行為,減少對復(fù)雜 JavaScript 和整頁重新加載的需求,從而使用戶體驗(yàn)更流暢、更具交互性。

          作為額外的好處,該應(yīng)用程序無需 JavaScript 也能正常運(yùn)行。

          然而,雖然我們的應(yīng)用程序功能齊全,但不可否認(rèn)它還有些簡陋。如果您希望繼續(xù)探索 htmx,您可能希望考慮在應(yīng)用程序狀態(tài)之間實(shí)現(xiàn)視圖轉(zhuǎn)換,或者向表單添加一些進(jìn)一步的驗(yàn)證 - 例如,驗(yàn)證電子郵件地址是否來自特定域。

          我在 htmx 簡介[8]中提供了這兩件事(以及更多)的示例。


          https://www.sitepoint.com/node-js-htmx-build-full-stack-app

          參考資料 [1]

          GitHub 存儲庫: https://github.com/jameshibbard/node-htmx

          [2]

          使用 Node 構(gòu)建簡單的初學(xué)者應(yīng)用程序: https://www.sitepoint.com/build-simple-beginner-app-node-bootstrap-mongodb/

          [3]

          Pug HTML 模板預(yù)處理器指南: https://www.sitepoint.com/a-beginners-guide-to-pug/

          [4]

          htmx 簡介: https://www.sitepoint.com/htmx-introduction/

          [5]

          GitHub 存儲庫中的 CSS 文件: https://github.com/jameshibbard/node-htmx/blob/main/public/styles.css

          [6]

          更多: https://htmx.org/attributes/hx-on/

          [7]

          express-validator: https://www.npmjs.com/package/express-validator

          [8]

          htmx 簡介: https://www.sitepoint.com/htmx-introduction/


          瀏覽 36
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  99免费在线视频 | 东京热二区| 国产成人无码精品A级毛片抽搐 | 日韩熟女色情视频一区二区三区 | 女人久久免费视频 |