本教程将指导您如何在node.js和ejs应用中实现无刷新动态搜索功能。通过利用javascript的dom事件监听和fetch api进行异步请求,我们将优化后端控制器以返回json数据,并在前端实时更新搜索结果,彻底解决传统表单提交导致的页面重载问题,显著提升用户体验。
在现代Web应用中,用户期望在输入搜索关键词或调整筛选条件时,能够即时看到结果更新,而无需点击提交按钮或等待页面刷新。这正是动态搜索的核心价值。本教程将针对一个Node.js Express应用结合EJS模板引擎的场景,详细阐述如何构建一个高效、响应式的动态搜索系统。
在提供的代码中,存在两个主要问题,导致动态搜索功能未能按预期工作:
为了解决这些问题,我们需要对前端和后端进行相应的改造。
为了支持前端的异步请求,我们需要一个能够返回JSON格式搜索结果的后端API端点。我们可以修改现有的getHome函数,使其能够根据请求类型(是普通页面加载还是AJAX请求)返回不同的响应,或者创建一个新的专用API函数。考虑到清晰性和职责分离,我们推荐创建一个新的API函数来处理AJAX请求。
user.controllers.js 改造:
我们将保留getHome用于初始页面加载和渲染EJS,并创建一个getSearchResultsAPI函数来专门处理前端的fetch请求。
const { pool } = require('../config/database.config');
const MiniSearch = require('minisearch');
const userModels = require('../models/user.models'); // 假设 formatDate 在这里
// 辅助函数:执行搜索和过滤逻辑
const performSearchAndFilter = async (searchedValue, deptFilter, yearFilter, fromDate, toDate) => {
const all_results = [];
const client = await pool.connect();
try {
const query = `select * from "users"`;
const result = await client.query(query);
result.rows.forEach((row) => {
all_results.push(row);
});
} catch (err) {
console.error("Database query error:", err);
// 可以在这里抛出错误或返回空数组
return [];
} finally {
client.release(); // 确保释放客户端
}
const minisearch = new MiniSearch({
fields: ['id', 'name', 'description', 'dept', 'year', 'fromDate', 'toDate'],
storeFields: ['id', 'name', 'description', 'dept', 'year', 'fromDate', 'toDate'],
});
minisearch.addAll(all_results);
const filterCriteria = (result, filters) => {
return Object.entries(filters).every(([key, value]) => {
if (!value) {
return true;
}
if (key === 'fromDate') {
const formattedDate = userModels.formatDate(result.fromDate);
return value <= formattedDate;
}
if (key === 'toDate') {
const formattedDate = userModels.formatDate(result.toDate);
return value >= formattedDate;
}
return result[key] !== undefined && result[key] === value;
});
};
const filters = {
dept: deptFilter,
year: yearFilter,
fromDate: fromDate,
toDate: toDate,
};
let results = [];
if (searchedValue) {
results = minisearch.search(searchedValue, {
prefix: true,
fuzzy: 0.4,
filter: (result) => filterCriteria(result, filters),
});
} else {
results = all_results.filter((result) => filterCriteria(result, filters));
}
return results.map((result) => ({
id: result.id,
name: result.name,
description: result.description,
}));
};
// 首页渲染函数 (保持不变,用于首次加载页面)
const getHome = async (req, res) => {
const searchedValue = req.query.searchedValue || '';
const deptFilter = req.query.deptFilter || '';
const yearFilter = req.query.yearFilter || '';
const fromDate = req.query.fromDate || '';
const toDate = req.query.toDate || '';
const search_results = await performSearchAndFilter(searchedValue, deptFilter, yearFilter, fromDate, toDate);
// 渲染EJS模板,包含初始或刷新后的搜索结果
res.render('../Views/user.ejs', {
search_results: search_results,
// 传递当前筛选值,以便前端可以回显
currentFilters: { searchedValue, deptFilter, yearFilter, fromDate, toDate }
});
};
// 新增的API函数,用于处理AJAX请求,返回JSON数据
const getSearchResultsAPI = async (req, res) => {
try {
const searchedValue = req.query.searchedValue || '';
const deptFilter = req.query.deptFilter || '';
const yearFilter = req.query.yearFilter || '';
const fromDate = req.query.fromDate || '';
const toDate = req.query.toDate || '';
const search_results = await performSearchAndFilter(searchedValue, deptFilter, yearFilter, fromDate, toDate);
// 直接返回JSON数据
res.json({ search_results: search_results });
} catch (error) {
console.error("API search error:", error);
res.status(500).json({ error: "Internal Server Error" });
}
};
module.exports = {
getHome,
getSearchResultsAPI, // 导出新的API函数
};路由配置 (app.js 或 routes.js):
确保您的Express应用中,/ 路由映射到getHome,而/search(或/api/search)路由映射到getSearchResultsAPI。
// 示例 Express 路由配置
const express = require('express');
const router = express.Router();
const userController = require('./controllers/user.controllers');
router.get('/', userController.getHome); // 初始页面加载
router.get('/search', userController.getSearchResultsAPI); // AJAX请求
module.exports = router;前端的主要任务是移除导致页面重载的表单提交,并确保JavaScript能够监听所有相关输入的变化,然后通过fetch API向新的后端API端点发送请求,并将返回的JSON数据动态渲染到页面上。
user.ejs 改造:
User
SEARCH HERE
<% if(search_results && search_results.length > 0) { %>
<% search_results.forEach((result) => { %>
<%= result.id %>
<%= result.name %>
<%= result.description %>
<% }) %>
<% } else { %>
No results found
<% } %>
通过上述改造,您的Node.js和EJS应用将拥有一个功能完善、响应迅速的动态搜索功能,大大提升用户体验。