精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

使用 Node.js 和 htmx 構建全棧 CRUD 應用程序

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

htmx 是一個現代 JavaScript 庫,旨在通過實現部分 HTML 更新來增強 Web 應用,而無需重新加載整個頁面。與傳統前端框架中的 JSON 有效載荷不同,它通過有線方式發送 HTML 來實現這一功能。

我們將要構建什么

我們將開發一個簡單的聯系人管理器,能夠執行所有 CRUD 操作:創建、讀取、更新和刪除聯系人。通過利用 htmx,該應用程序將提供單頁應用程序 (SPA) 的感覺,從而增強交互性和用戶體驗。

如果用戶禁用 JavaScript,應用程序將以整頁刷新的方式運行,從而保持可用性和可發現性。這種方法展示了 htmx 創建現代 Web 應用程序的能力,同時保持它們的可訪問性和 SEO 友好性。

這就是我們最終得到的結果。

圖片圖片

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

先決條件

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

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

在開始之前,請運行以下命令:

node -v
npm -v

您應該看到如下輸出:

v20.11.1
10.4.0

這確認了 Node 和 npm 已安裝在您的計算機上,并且可以從命令行環境進行訪問。

設置項目

讓我們從搭建一個新的 Node 項目開始:

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

這應該在項目根目錄中創建一個 package.json 文件。

接下來,讓我們安裝我們需要的依賴項:

npm i express method-override pug

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

接下來,在根目錄中創建一個 app.js 文件:

touch app.js

并添加以下內容:

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}`);
});

在這里,我們正在設置 Express 應用程序的結構。這包括將 Pug 配置為渲染視圖的視圖引擎、定義靜態資產的目錄以及連接路由器。

該應用程序偵聽端口 3000,并使用控制臺日志來確認 Express 正在運行并準備好處理指定端口上的請求。此設置構成了我們應用程序的基礎,并準備好通過更多功能和路由進行擴展。

接下來,讓我們創建路由文件:

mkdir routes
touch routes/index.js

打開該文件并添加以下內容:

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

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

在這里,我們在新創建的路由目錄中設置基本路由。此路由在 /contacts 端點偵聽 GET 請求,并使用簡單的確認消息進行響應,表明一切正常。

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

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

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

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

圖片圖片

激動人心的時刻!

顯示所有聯系人

現在讓我們添加一些要顯示的聯系人。由于我們專注于 htmx,因此為了簡單起見,我們將使用硬編碼數組。這將使事情變得精簡,使我們能夠專注于 htmx 的動態功能,而無需復雜的數據庫集成。

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

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

const contacts = [
  { id: 1, name: 'John Doe', email: 'john.doe@example.com' },
  { id: 2, name: 'Jane Smith', email: 'jane.smith@example.com' },
  { id: 3, name: 'Emily Johnson', email: 'emily.johnson@example.com' },
  { id: 4, name: 'Aarav Patel', email: 'aarav.patel@example.com' },
  { id: 5, name: 'Liu Wei', email: 'liu.wei@example.com' },
  { id: 6, name: 'Fatima Zahra', email: 'fatima.zahra@example.com' },
  { id: 7, name: 'Carlos Hernández', email: 'carlos.hernandez@example.com' },
  { id: 8, name: 'Olivia Kim', email: 'olivia.kim@example.com' },
  { id: 9, name: 'Kwame Nkrumah', email: 'kwame.nkrumah@example.com' },
  { id: 10, name: 'Chen Yu', email: 'chen.yu@example.com' },
];

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

mkdir views
touch views/index.pug

并添加以下內容:

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/htmx.org@1.9.10')

在此模板中,我們為應用程序布置 HTML 結構。在 head 部分,我們包含了來自 Google Fonts 的 Roboto 字體和自定義樣式的樣式表。

正文分為標題、用于列出聯系人的側邊欄以及用于存放所有聯系信息的主要內容區域。內容區域當前包含一個占位符。在正文的末尾,我們還包含來自 CDN 的最新版本的 htmx 庫。

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

接下來,讓我們創建自定義樣式表:

mkdir public
touch public/styles.css

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

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

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

現在,當您刷新頁面時,您應該會看到類似這樣的內容。

圖片圖片

顯示單個聯系人

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

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

touch views/sidebar.pug

將以下內容添加到這個新文件中:

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

這里我們為每個聯系人創建了一個指向 /contacts/${contact.id} 的鏈接,并添加了三個 htmx 屬性:

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

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

section#sidebar
  include sidebar.pug

請記?。篜ug 對空格敏感,因此請務必使用正確的縮進。

現在讓我們在 index.js 中創建一個新端點以返回 htmx 期望的 HTML 響應:

// 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>
  `);
});

如果保存并刷新瀏覽器,您現在應該能夠查看每個聯系人的詳細信息。

圖片圖片

網絡上的 HTML

讓我們花點時間了解一下這里發生了什么。正如文章開頭提到的,htmx 通過網絡傳輸 HTML,而不是傳統前端框架的 JSON 有效負載。

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

圖片圖片

處理全頁刷新

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

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

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

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

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

// 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 });
  }
});

現在,當您重新加載頁面時,用戶界面不會消失。但是,它確實會從您正在查看的任何聯系人恢復為消息“選擇聯系人”,這并不理想。

為了解決這個問題,我們可以在 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 });
}

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

這樣,我們的應用程序應該能夠承受刷新或直接加載聯系人。

快速重構

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

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

touch views/contact.pug

在新創建的模板中,添加以下內容:

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 });
}

事情應該仍然像以前一樣工作,但現在我們已經刪除了重復的代碼。

新的聯系表

我們要關注的下一個任務是創建新聯系人。本教程的這一部分將指導您設置表單和后端邏輯,使用 htmx 動態處理提交。

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

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 屬性來顯示聯系人:hx-get 將通過 Ajax 向 /contacts/new 端點發出 GET 請求,hx-target 將指定插入響應的位置,hx-push-url 將確保更改 URL。

現在讓我們為表單創建一個新模板:

touch views/form.pug

并添加以下代碼:

h2 New Contact

form(
  actinotallow='/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 端點發出帶有表單數據的 POST 請求。結果(更新的聯系人列表)將被插入到側邊欄中。在這種情況下,我們不想更改 URL,因為用戶可能想要輸入多個新聯系人。但是,我們確實希望在成功提交后清空表單,這就是 hx-on::after-request 的作用。 hx-on* 屬性允許您內聯嵌入腳本以直接響應元素上的事件。你可以在這里讀更多[6]關于它的內容。

接下來,我們在 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 的聯系人。

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

when 'new'
  include form.pug

刷新頁面,此時您應該能夠通過單擊側欄中的“New Contact”鏈接來呈現新的聯系人表單。

圖片圖片

創建聯系人

現在我們需要創建一個路由來處理表單提交。

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

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({ extended: true }));
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 包,但我最近了解到這不再是必要的。

然后將以下內容添加到 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: {} });
  }
});

在這里,我們使用從客戶端收到的數據創建一個新聯系人,并將其添加到 contacts 數組中。然后,我們重新渲染側邊欄,并向其傳遞更新的聯系人列表。

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

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

現在,如果您刷新瀏覽器并嘗試添加聯系人,它應該按預期工作:新聯系人應添加到側邊欄,并且應重置表單。

添加 toast 消息提示

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

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

為了解決這個問題,我們可以使用 hx-swap-oob 屬性。這允許您指定響應中的某些內容應交換到目標以外的 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: {} });
}

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

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

現在,當我們添加聯系人時,我們應該會收到一條不錯的消息,告訴我們發生了什么。

圖片圖片

編輯聯系人

為了更新聯系人,我們將重用上一節中創建的表單。

讓我們首先更新 contact.pug 模板以添加以下內容:

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

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

現在讓我們更改 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 檢索聯系人,然后將該聯系人傳遞到表單。

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

// 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(
  actinotallow=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

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

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

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

這意味著:

form(
  actinotallow=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
)

當 isEditing() 返回 false 時將編譯為以下內容:

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

但是當 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>

在其更新狀態下,請注意表單操作是 "/update/1?_method=PUT" 。添加此查詢字符串參數是因為我們正在使用方法覆蓋包,它將使我們的路由器響應 PUT 請求。

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

讓我們繼續將其添加到 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({ extended: true }));
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 = {
    id: Number(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}`);
  }
});

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

在處理 htmx 請求時,我們首先用更新的聯系人列表呈現側邊欄模板。然后,我們用更新的聯系人渲染聯系人模板,并使用這兩次調用的結果來組合我們的響應。與之前一樣,我們使用 “Out of Band” 更新創建一條 toast 消息,告知用戶聯系人已更新。

此時,您應該能夠更新聯系人。

刪除聯系人

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

div.actions
  form(method='POST', actinotallow=`/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
  )

請注意,最好使用表單和按鈕來發出 DELETE 請求。表單是為導致更改(例如刪除)的操作而設計的,這確保了語義的正確性。此外,使用鏈接進行刪除操作可能存在風險,因為搜索引擎可能會無意中跟蹤鏈接,從而可能導致不必要的刪除。

話雖這么說,我添加了一些 CSS 來將按鈕設置為鏈接樣式,因為按鈕很難看。如果您之前從存儲庫復制了樣式,那么您的代碼中已經包含了該樣式。

最后,我們的路由處理程序在 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');
  }
});

刪除聯系人后,我們將更新側邊欄并向用戶顯示一條提示消息。

圖片圖片

更進一步

就這樣吧。

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

作為額外的好處,該應用程序無需 JavaScript 也能正常運行。

然而,雖然我們的應用程序功能齊全,但不可否認它還有些簡陋。如果您希望繼續探索 htmx,您可能希望考慮在應用程序狀態之間實現視圖轉換,或者向表單添加一些進一步的驗證 - 例如,驗證電子郵件地址是否來自特定域。

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

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

責任編輯:武曉燕 來源: 獨立開發者張張
相關推薦

2022-05-09 17:33:23

PWA漸進式Web應用程序離線優先

2024-03-27 11:18:02

2013-03-28 14:54:36

2021-06-15 15:03:21

MongoDBNode.jsCRUD

2020-09-04 15:06:04

Docker容器化Node.js

2013-05-17 09:41:02

Node.js云應用開發IaaS

2020-09-22 07:35:42

Node.jsVue.js文件壓縮

2023-01-10 14:11:26

2024-12-24 08:12:59

2022-09-12 16:02:32

Docker安全Node.js

2023-03-07 14:31:44

Node.jsPython應用程序

2023-04-18 15:18:10

2022-08-22 07:26:32

Node.js微服務架構

2020-01-15 14:20:07

Node.js應用程序javascript

2022-12-14 14:40:27

Node.js開發應用程序

2022-09-12 15:58:50

node.js微服務Web

2012-09-29 11:13:15

Node.JS前端開發Node.js打包

2012-09-17 11:26:14

IBMdw

2023-10-26 01:28:02

2015-11-20 17:09:36

jsWeb應用程序
點贊
收藏

51CTO技術棧公眾號

免费观看一区二区三区| 黄黄视频在线观看| 中文字幕+乱码+中文| 香蕉视频国产精品| 亚洲国产天堂久久综合网| 天天影视综合色| 性欧美videos高清hd4k| 91蝌蚪porny成人天涯| 国产欧美精品一区二区三区-老狼| 欧美日韩大片在线观看| 欧洲福利电影| 精品卡一卡二卡三卡四在线| 99热手机在线| 免费h视频在线观看| 中文字幕一区二区三区四区| 精品一卡二卡三卡四卡日本乱码| 国产孕妇孕交大片孕| 国产精品综合色区在线观看| 久久综合免费视频| 亚洲精品乱码久久久久久久久久久久 | 亚洲精品一区中文字幕乱码| 夜夜爽久久精品91| 视频一区在线免费看| 午夜精品视频一区| 美女黄色片网站| www亚洲人| 91蜜桃传媒精品久久久一区二区| 91亚洲一区精品| 亚洲 小说区 图片区| 99精品福利视频| 欧美黄色免费网站| 疯狂撞击丝袜人妻| 成人影院天天5g天天爽无毒影院| 亚洲精品aⅴ中文字幕乱码| 熟妇无码乱子成人精品| 视频欧美精品| 欧美日韩中文另类| www.欧美日本| 中文字幕日本一区二区| 欧美视频中文在线看| 很污的网站在线观看| 天堂va在线| 一区二区三区色| 黑人巨茎大战欧美白妇| av网址在线看| 亚洲视频中文字幕| 欧美性受xxxx黑人猛交88| 日本在线天堂| 亚洲日本丝袜连裤袜办公室| 一区二区三区四区久久| 午夜视频在线免费观看| 国产精品欧美精品| 亚洲视频在线二区| 男人资源在线播放| 亚洲人精品午夜| 91麻豆天美传媒在线| 成人影院在线观看| 一区二区三区在线免费播放| 日本黄网站色大片免费观看| 亚洲淫性视频| 亚洲成人一区在线| 99热自拍偷拍| 婷婷午夜社区一区| 欧美午夜寂寞影院| 天天插天天操天天射| 国产日本久久| 欧美一级免费大片| 国产情侣久久久久aⅴ免费| 欧美黑人巨大videos精品| 精品亚洲一区二区三区| 免费毛片视频网站| 99九九热只有国产精品| 欧美黄色www| 日韩欧美成人一区二区三区 | 日本中文字幕二区| 精品伊人久久| 亚洲高清色综合| 久久精品视频18| 欧美gay男男猛男无套| 超碰日本道色综合久久综合| 久久综合成人网| 欧美专区18| 国产在线视频一区| 亚洲女同志亚洲女同女播放| 久久亚洲春色中文字幕久久久| 日韩精品久久久| 伊人影院在线视频| 欧美日韩在线免费| 日韩av自拍偷拍| 香蕉久久99| 日韩有码在线观看| 日韩欧美a级片| 美洲天堂一区二卡三卡四卡视频| 97视频中文字幕| 成人精品一区| 亚洲电影一级黄| 一级片视频免费观看| swag国产精品一区二区| 在线亚洲男人天堂| 日韩成人av毛片| 久久99蜜桃精品| 免费在线观看一区二区| av软件在线观看| 欧美亚洲日本国产| av网站有哪些| 欧美在线免费一级片| 国产高清在线不卡| 欧美综合视频在线| 中文字幕日韩一区| 久久9精品区-无套内射无码| 视频在线观看免费影院欧美meiju| 亚洲精品自拍第一页| 免费看一级一片| 美女国产一区二区| 欧美日韩一区二区三| 久久99亚洲网美利坚合众国| 欧美日本高清视频在线观看| 色婷婷免费视频| 欧美一区二区| 国产伦精品免费视频| 婷婷国产在线| 亚洲成av人片在线| 成人高清在线观看视频| 四季av一区二区凹凸精品| 热久久99这里有精品| 日韩永久免费视频| 一区二区高清免费观看影视大全| 中日韩av在线播放| 欧美亚洲精品在线| 国产99在线|中文| 台湾av在线二三区观看| 亚洲va欧美va天堂v国产综合| 日本黄色www| 我不卡影院28| 91午夜理伦私人影院| 尤物网址在线观看| 欧美日韩在线播放| 国产精品1区2区3区4区| 日av在线不卡| 亚洲 国产 日韩 综合一区| 日韩天堂在线| 亚洲美女在线观看| 69视频免费看| 国产亚洲欧美日韩俺去了| 六月丁香婷婷在线| 美女久久久久| 国产精品69久久| 国产区在线视频| 在线观看视频一区二区| 久久久久久国产精品无码| 久久xxxx精品视频| 日日噜噜噜噜夜夜爽亚洲精品| 欧美极品免费| 在线看片第一页欧美| 在线免费观看中文字幕| 亚洲色图一区二区三区| 精品国产鲁一鲁一区二区三区| 91高清一区| 97自拍视频| av人人综合网| 国产亚洲aⅴaaaaaa毛片| 国产一卡二卡三卡| 亚洲素人一区二区| jjzz黄色片| 亚洲免费影院| 亚洲aⅴ天堂av在线电影软件| 97精品国产99久久久久久免费| 中文字幕不卡在线视频极品| 7777久久亚洲中文字幕| 亚洲精品中文字幕在线观看| 亚洲图片欧美另类| 羞羞答答国产精品www一本| 青娱乐一区二区| 成人精品视频在线观看| 欧美激情aaaa| 邻居大乳一区二区三区| 欧美老年两性高潮| 国产主播在线播放| 中文字幕乱码一区二区免费| 99精品视频国产| 在线视频精品| 亚洲日本欧美在线| 国产精品毛片久久久| 国产精品爱久久久久久久| 成年视频在线观看| 日韩成人av在线| 97超碰资源站| 欧美日韩亚洲成人| 日韩精品123区| 91免费在线播放| 91精产国品一二三产区别沈先生| 精品成人国产| 亚洲精品中文综合第一页| eeuss国产一区二区三区四区| 日韩av高清不卡| 少女频道在线观看高清| 国产亚洲欧美aaaa| 日日躁夜夜躁白天躁晚上躁91| 欧美午夜影院在线视频| 2025国产精品自拍| 久久久国产综合精品女国产盗摄| 樱花草www在线| 久久久久久久高潮| 亚洲人成无码网站久久99热国产| 欧美日韩在线二区| 精品人伦一区二区三区| 白嫩亚洲一区二区三区| 国产成人一区三区| 97人人爽人人澡人人精品| 色偷偷噜噜噜亚洲男人的天堂| 污污的视频网站在线观看| 欧美日韩一二三| 在线观看黄网站| 亚洲国产美国国产综合一区二区| 91免费在线看片| 久久久91精品国产一区二区精品| 午夜福利三级理论电影 | 国产中文欧美精品| 日韩三区在线| 日本电影亚洲天堂| 999av小视频在线| 欧美成人网在线| 免费看a在线观看| 伊人激情综合网| 韩国三级在线观看久| 亚洲精品国产欧美| 少妇荡乳情欲办公室456视频| 欧美一级日韩不卡播放免费| 亚洲天堂久久久久| 欧美视频一区二区在线观看| 激情视频网站在线观看| 精品国产精品自拍| 国产手机在线视频| 五月婷婷久久丁香| 日本少妇性生活| 亚洲专区一二三| 久久黄色免费视频| 亚洲欧美二区三区| www日韩在线| 日韩码欧中文字| 国产午夜精品理论片在线| 中文字幕视频一区| 秋霞欧美一区二区三区视频免费| 中文在线资源观看网站视频免费不卡| 熟女俱乐部一区二区视频在线| 久久综合久久综合久久| 爱爱免费小视频| 国产三级精品三级| 林心如三级全黄裸体| 亚洲欧洲av一区二区三区久久| 亚洲色偷偷综合亚洲av伊人| ●精品国产综合乱码久久久久| 潘金莲一级黄色片| 亚洲综合视频网| 日韩精品一区二区不卡| 欧美日韩午夜激情| 中文在线资源天堂| 欧美一区二区在线播放| 国产肥老妇视频| 亚洲国产精品久久久久| 亚洲欧洲精品视频| 在线日韩第一页| 国产成人l区| 久久免费国产精品1| 女人让男人操自己视频在线观看 | 中文字幕久久精品一区二区| 国产精品加勒比| 亚洲福利天堂| 一区二区不卡在线观看| 午夜精品网站| 91精品91久久久中77777老牛 | 日本少妇激三级做爰在线| 国产黄色成人av| 91视频啊啊啊| 国产精品女同一区二区三区| 校园春色 亚洲| 欧美日韩亚洲国产一区| 在线观看日韩一区二区| 精品卡一卡二卡三卡四在线| 久草在线免费福利资源| 久久人人爽人人爽爽久久 | 午夜精品一区在线观看| 99精品在线播放| 制服丝袜成人动漫| 亚洲欧美日韩免费| 久久天天躁狠狠躁夜夜躁| 波多野在线观看| 国产精品美乳在线观看| 中文字幕区一区二区三| 日韩中文不卡| 亚洲福利久久| 九九九九九国产| 99精品国产99久久久久久白柏| 亚洲女人毛茸茸高潮| 亚洲风情在线资源站| 中文字幕在线播放日韩| 亚洲国产成人精品久久久国产成人一区| 国产小视频福利在线| 欧美激情国产日韩精品一区18| 日韩精品一区二区三区| 91丝袜脚交足在线播放| 日韩在线欧美| 大陆极品少妇内射aaaaa| 韩国一区二区在线观看| 熟女高潮一区二区三区| 亚洲午夜免费视频| 国产精品爽爽久久| 国产午夜精品一区二区三区 | 国产永久免费高清在线观看| 欧美激情乱人伦一区| 九七影院97影院理论片久久 | 极品美乳网红视频免费在线观看| 欧美二区在线播放| 香蕉久久一区| 亚洲在线欧美| 日本特黄久久久高潮| 人妻熟女aⅴ一区二区三区汇编| 一区二区三区视频在线观看| 91精品中文字幕| 尤物tv国产一区| 中文字幕 在线观看| 国产精品一区二区欧美| 欧美午夜不卡| 性一交一黄一片| 亚洲丝袜制服诱惑| 97视频免费在线| 深夜成人在线观看| 成人18视频在线观看| 日本一区免费| 久久裸体视频| www.色天使| 色综合色狠狠综合色| 亚洲人视频在线观看| 91精品国产自产91精品| 久久99国产精品久久99大师| 丁香婷婷综合激情| 成人免费观看视频| 久久亚洲AV无码| 亚洲国产精品va在线| 国产盗摄精品一区二区酒店| 国产精品久久久久久久久久久久冷 | 久久夜色精品国产欧美乱| 欧美啪啪网站| 在线免费观看成人| 国内精品伊人久久久久av影院 | 亚欧美无遮挡hd高清在线视频| 激情五月俺来也| 亚洲天堂a在线| 国产黄频在线观看| 欧美激情国产高清| 鲁大师精品99久久久| 国产性xxxx18免费观看视频| 久久一区二区三区四区| 天天爽夜夜爽人人爽| 在线观看国产成人av片| **国产精品| 日韩精品一区二区免费| av影院午夜一区| 国产99免费视频| 色综久久综合桃花网| 视频精品一区二区三区| 欧美精品久久久久久久自慰| 91免费看片在线观看| 欧美三级网站在线观看| www.日韩不卡电影av| 99精品在免费线中文字幕网站一区 | 在线观看福利电影| 先锋影音一区二区三区| 国内精品伊人久久久久av一坑| 久久免费少妇高潮99精品| 亚洲精品mp4| 日韩亚洲国产免费| 妺妺窝人体色777777| 久久亚区不卡日本| 国产精品欧美久久久久天天影视 | 国产无人区码熟妇毛片多| 中文字幕国产亚洲2019| 无码国模国产在线观看| 免费在线观看的av网站| 自拍偷拍欧美激情| 午夜视频www| 成人国产精品久久久久久亚洲| 亚洲三级免费| 国产三级在线观看完整版| 日韩精品一区二区三区swag | 91久久国产综合久久91精品网站 | 69久久夜色精品国产69| 日韩在线综合| 亚洲国产综合视频| 91精品国产综合久久久蜜臀图片| 欧美freesex黑人又粗又大| 在线亚洲美日韩| 91网址在线看| 精品久久久久中文慕人妻| 国产福利精品视频| 最新亚洲视频| 少妇被躁爽到高潮无码文| 亚洲色图综合网| 91午夜精品| 男生操女生视频在线观看 |