前端程式碼
前端程式碼
<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="stylesheet" href="./css/index.css">
<link rel="stylesheet" href="./css/prism.css">
<title>blog</title>
<style>
[v-cloak] {
display: none;
}
</style>
</head>
<body>
<div id="app">
<!-- 移動端選單 -->
<span class="mnavbtn" onclick="openMnav()">
<svg width="25" height="25" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 448 512">
<path
d="M0 96C0 78.3 14.3 64 32 64H416c17.7 0 32 14.3 32 32s-14.3 32-32 32H32C14.3 128 0 113.7 0 96zM0 256c0-17.7 14.3-32 32-32H416c17.7 0 32 14.3 32 32s-14.3 32-32 32H32c-17.7 0-32-14.3-32-32zM448 416c0 17.7-14.3 32-32 32H32c-17.7 0-32-14.3-32-32s14.3-32 32-32H416c17.7 0 32 14.3 32 32z" />
</svg>
</span>
<div id="mySideMnav" class="sidemnav">
<a href="javascript:;" class="closemnavbtn" onclick="closeMnav()">×</a>
<a href="#">首頁</a>
<a href="#">歸檔</a>
<a href="#">關於</a>
<a href="#">資源</a>
</div>
<div class="container">
<!-- 左側導航 -->
<div class="nav">
<div class="logo">
<img src="./img/logo.jpg">
<h2>※聽雨※</h2>
</div>
<div style="margin: 30px auto;text-align: center;">
<span style="cursor: pointer;margin: 0 5px;">
<svg width="25" height="25" xmlns="http://www.w3.org/2000/svg" width="16" height="16"
fill="currentColor" class="bi bi-envelope-fill" viewBox="0 0 16 16">
<path
d="M.05 3.555A2 2 0 0 1 2 2h12a2 2 0 0 1 1.95 1.555L8 8.414.05 3.555ZM0 4.697v7.104l5.803-3.558L0 4.697ZM6.761 8.83l-6.57 4.027A2 2 0 0 0 2 14h12a2 2 0 0 0 1.808-1.144l-6.57-4.027L8 9.586l-1.239-.757Zm3.436-.586L16 11.801V4.697l-5.803 3.546Z" />
</svg>
</span>
<span style="cursor: pointer;margin: 0 5px;">
<svg width="25" height="25" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 496 512">
<path
d="M165.9 397.4c0 2-2.3 3.6-5.2 3.6-3.3.3-5.6-1.3-5.6-3.6 0-2 2.3-3.6 5.2-3.6 3-.3 5.6 1.3 5.6 3.6zm-31.1-4.5c-.7 2 1.3 4.3 4.3 4.9 2.6 1 5.6 0 6.2-2s-1.3-4.3-4.3-5.2c-2.6-.7-5.5.3-6.2 2.3zm44.2-1.7c-2.9.7-4.9 2.6-4.6 4.9.3 2 2.9 3.3 5.9 2.6 2.9-.7 4.9-2.6 4.6-4.6-.3-1.9-3-3.2-5.9-2.9zM244.8 8C106.1 8 0 113.3 0 252c0 110.9 69.8 205.8 169.5 239.2 12.8 2.3 17.3-5.6 17.3-12.1 0-6.2-.3-40.4-.3-61.4 0 0-70 15-84.7-29.8 0 0-11.4-29.1-27.8-36.6 0 0-22.9-15.7 1.6-15.4 0 0 24.9 2 38.6 25.8 21.9 38.6 58.6 27.5 72.9 20.9 2.3-16 8.8-27.1 16-33.7-55.9-6.2-112.3-14.3-112.3-110.5 0-27.5 7.6-41.3 23.6-58.9-2.6-6.5-11.1-33.3 2.6-67.9 20.9-6.5 69 27 69 27 20-5.6 41.5-8.5 62.8-8.5s42.8 2.9 62.8 8.5c0 0 48.1-33.6 69-27 13.7 34.7 5.2 61.4 2.6 67.9 16 17.7 25.8 31.5 25.8 58.9 0 96.5-58.9 104.2-114.8 110.5 9.2 7.9 17 22.9 17 46.4 0 33.7-.3 75.4-.3 83.6 0 6.5 4.6 14.4 17.3 12.1C428.2 457.8 496 362.9 496 252 496 113.3 383.5 8 244.8 8zM97.2 352.9c-1.3 1-1 3.3.7 5.2 1.6 1.6 3.9 2.3 5.2 1 1.3-1 1-3.3-.7-5.2-1.6-1.6-3.9-2.3-5.2-1zm-10.8-8.1c-.7 1.3.3 2.9 2.3 3.9 1.6 1 3.6.7 4.3-.7.7-1.3-.3-2.9-2.3-3.9-2-.6-3.6-.3-4.3.7zm32.4 35.6c-1.6 1.3-1 4.3 1.3 6.2 2.3 2.3 5.2 2.6 6.5 1 1.3-1.3.7-4.3-1.3-6.2-2.2-2.3-5.2-2.6-6.5-1zm-11.4-14.7c-1.6 1-1.6 3.6 0 5.9 1.6 2.3 4.3 3.3 5.6 2.3 1.6-1.3 1.6-3.9 0-6.2-1.4-2.3-4-3.3-5.6-2z" />
</svg>
</span>
</div>
<div class="navlink">
<ul class="navlist">
<li>
<a href="/">首頁</a>
</li>
<li>
<a href="javascript:;">歸檔</a>
</li>
<li>
<a href="javascript:;">關於</a>
</li>
<li>
<a href="javascript:;">資源</a>
</li>
</ul>
</div>
</div>
<!-- 中間內容 -->
<div class="main">
<div style="margin-top: 35px;"></div>
<div v-cloak v-if="articleListIsShow">
<template v-for="(article, index) in articleList" :key="index">
<div class="article-card" :class="{sticky: article.istop}">
<div class="article-card-container">
<h4 class="article-card-title" @click="getArticleDetail(article.id)">
<i class="sticky-icon" v-cloak v-if="article.istop">⚘</i>
{{article.title}}
</h4>
<div class="article-card-des">{{article.description}}</div>
<div class="article-card-footer">
<span class="article-card-footer-author">作者:聽雨</span>
<span class="article-card-footer-date">釋出時間:{{article.created_at}}</span>
<!-- <span class="article-card-footer-cate">分類</span> -->
</div>
</div>
</div>
</template>
</div>
<!-- 文章詳情 -->
<div class="article-container" v-cloak v-if="articleDetailIsShow">
<div style="float: right;color: red;cursor: pointer;" @click="returnHomePage()"><<返回</div>
<div class="article-main">
<article class="article-view">
<h1 class="article-title">{{article.title}}</h1>
<div class="article-meta">
<span>作者: 聽雨</span>
<!-- <span>
分類:
<span>Linux</span>
</span> -->
<span>釋出時間: {{article.created_at}}</span>
</div>
<div class="article-desc">{{article.description}}</div>
<div class="article-content" v-html="article.content"></div>
</article>
</div>
</div>
</div>
<!-- 右側皮膚 -->
<div class="rside" v-cloak v-if="rsideIsShow">
<!-- 搜尋框 -->
<div class="search-box">
<input class="search-txt" type="text" placeholder="輸入內容搜尋">
<input class="search-bnt" type="submit" value="搜尋">
</div>
<!-- 文章分類 -->
<div class="rside-card">
<div class="rside-card-container">
<h4 class="rside-card-title">分類</h4>
<div class="rside-card-body">
<li v-for="(cate, index) in cateList" :key="index">{{cate.name}}</li>
</div>
</div>
</div>
<!-- 文章標籤 -->
<!-- <div class="rside-card">
<div class="rside-card-container">
<h4 class="rside-card-title">標籤</h4>
<div class="rside-card-body">
<li>winform</li>
<li>spring</li>
<li>Django</li>
</div>
</div>
</div> -->
</div>
</div>
</div>
<script src="./js/index.js"></script>
<script src="./js/zepto.min.js"></script>
<script src="./js/prism.js"></script>
<script type="module">
import { createApp, ref, onMounted, } from "./js/vue.esm-browser.prod.js";
createApp({
setup() {
const articleList = ref({}); // 首頁文章列表
const cateList = ref({}); // 首頁分類列表
const article = ref({}); // 文章詳情物件
const articleListIsShow = ref(true); // 是否顯示文章列表
const articleDetailIsShow = ref(false); // 是否顯示文章詳情
const rsideIsShow = ref(true); // 是否顯示右側皮膚
// 獲取首頁文章列表
const getArticleList = () => {
$.ajax({
url: 'http://localhost:5588/api/home/articles',
type: 'GET',
dataType: 'json',
success: function (res) {
if (res.code == 200) {
let data = res.data.sort((a, b) => b.istop - a.istop);
articleList.value = data;
}
},
error: function (err) {
console.log(err.responseText);
}
});
};
// 獲取首頁分類列表
const getCateList = () => {
$.ajax({
url: 'http://localhost:5588/api/home/cates',
type: 'GET',
dataType: 'json',
success: function (res) {
if (res.code == 200) {
cateList.value = res.data;
}
},
error: function (err) {
console.log(err.responseText);
}
});
};
// 獲取文章詳情
const getArticleDetail = (id) => {
$.ajax({
url: `http://localhost:5588/api/article/${id}`,
type: 'GET',
dataType: 'json',
success: function (res) {
if (res.code == 200) {
article.value = res.data;
rsideIsShow.value = false;
articleListIsShow.value = false;
articleDetailIsShow.value = true;
}
},
error: function (err) {
console.log(err.responseText);
}
});
};
// 從文章詳情頁返回首頁
const returnHomePage = () => {
articleDetailIsShow.value = false;
articleListIsShow.value = true;
rsideIsShow.value = true;
};
// 頁面載入時執行
onMounted(() => {
getArticleList();
getCateList();
});
return {
articleList,
cateList,
article,
articleListIsShow,
articleDetailIsShow,
rsideIsShow,
getArticleDetail,
returnHomePage,
}
}
}).mount('#app');
</script>
</body>
</html>
後端程式碼
// 文章詳情
router.get("/api/article/:id", &articleController.getArticleDetail);
article_controller.zig
中新增對應函式
/// 獲取文章詳情
pub fn getArticleDetail(req: *httpz.Request, res: *httpz.Response) !void {
const article_id = req.param("id").?;
if (article_id.len == 0) {
res.status = 400;
try res.json(.{ .code = 400, .msg = "錯誤的請求" }, .{});
return;
}
const id = try std.fmt.parseInt(u32, article_id, 10);
const article_op = try article_server.getArticleDetail(id);
if (article_op) |article| {
res.status = 200;
try res.json(.{ .code = 200, .msg = "ok", .data = article }, .{});
} else {
res.status = 404;
try res.json(.{ .code = 404, .msg = "文章不存在" }, .{});
return;
}
}
article_server.zig
中新增對應函式
/// 獲取文章詳情
pub fn getArticleDetail(id: u32) !?ArticleModel.Article {
var db = try database.OpenDb();
defer db.deinit();
const query =
\\SELECT id, title, description, content, istop, created_at, updated_at FROM article WHERE id = ?
;
var stmt = try db.prepare(query);
defer stmt.deinit();
var gpa = std.heap.GeneralPurposeAllocator(.{}){};
const allocator = gpa.allocator();
const row = try stmt.oneAlloc(ArticleModel.Article, allocator, .{}, .{
.id = id,
});
if (row) |r| {
return r;
} else {
return null;
}
}