首页 > 其他分享 >使用zig语言制作简单博客网站(六)文章详情页

使用zig语言制作简单博客网站(六)文章详情页

时间:2024-08-29 22:52:24浏览次数:9  
标签:const res 博客 try zig 详情页 value article id

前端代码

前端代码
<!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()">&times;</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()">&lt;&lt;返回</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;
    }
}

标签:const,res,博客,try,zig,详情页,value,article,id
From: https://www.cnblogs.com/tingyublog/p/18387674

相关文章

  • 我的第一个博客内容
    学习markdown的使用加粗一句话加粗是在一句话中各加两个星,好像双引号斜体一句话斜体是在一句话中各加一个星,好像单引号高亮一句话我在typora高亮失败了,在这里试一下上标试一下原来是在一个数字两边加^,22再来一个下标哈哈哈哈哈哈哈,下标就是在要下标的位置左右加~呢H2O......
  • 一分钟搭建Ghost个人网站博客系统
    什么是Ghost博客系统Ghost是一款设计简约、主题精致的个人博客系统,Ghost支持多用户创建和编辑,支持Markdown格式撰写文章,编辑的内容可即时预览。创建轻量云主机这里默认你已经有对应的轻量云服务器了。如果你已经有云服务器,直接重新安装,按照下面的流程就可以了。没有的的可以......
  • 计算机毕业设计django+vue个人博客系统【开题+论文+程序】
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表开题报告内容研究背景随着互联网技术的飞速发展,个人博客作为展示个人思想、分享知识与经验的重要平台,其重要性日益凸显。传统的博客系统多依赖于单一的技术栈,如......
  • 博客园-awescnb插件-geek皮肤优化--浏览器ico图标修改
    简介通过js方式自定义修改博客园-awescnb插件-geek皮肤下浏览器ico图标图标准备准备自定义的图标,上传至博客园个人的相册中代码注入定义自定义HTML:博客园->管理->设置->页脚HTML代码添加相关代码//更换ico图标functionupdateICO(){varlink=docum......
  • 用我十多年的“奇葩”经验,给在“挂吊瓶”的博客园几点建议
    初识博客园我是08年开始接触开发的,一开始涉及的就是.net和java,记得那会好像是jar6来着,net嘛还是2.0那时候包括现在,找资料很多时候会找到博客园来一开始我以为博客园是很多博主成立的一个联盟,就是各自弄一个博客系统,然后公用一个域名为啥会这么想呢?因为我看高的博文都长得不一......
  • 博客园-awescnb插件-geek皮肤优化--浏览器ico图标修改
    简介通过js方式自定义修改博客园-awescnb插件-geek皮肤下浏览器ico图标图标准备准备自定义的图标,上传至博客园个人的相册中代码注入定义自定义HTML:博客园->管理->设置->页脚HTML代码添加相关代码//更换ico图标functionupdateICO(){varlink=document.qu......
  • 博客园美化系列第四弹
    <!--鼠标点击特效--><scripttype="text/javascript">vara_idx=0;jQuery(document).ready(function($){$("body").click(function(e){vara=newArray("❤富强❤","❤民主❤","❤文明❤","❤和谐❤"......
  • 博客园美化系列第五弹
    <!--鼠标星星--><scripttype="text/javascript">if(screen&&screen.width>860){document.write('<scripttype="text/javascript"src="https://blog-static.cnblogs.com/files/vivin-echo/curs......
  • 博客园美化系列总结
    页面定制css代码//鼠标指针body{cursor:url('https://files-cdn.cnblogs.com/files/miluluyo/cursora.ico'),auto;background-color:whitesmoke;//修改背景颜色为半透明}//loading@keyframesspin3D{from{transform:rotate3d(0.5,0.5,0.5,360deg)}to{transfo......
  • 博客园美化系列第一弹
    博客园美化系列第一弹首先要确保你已经申请开通博客「理由随便写,积极向上即可」,且已通过审核。然后进入【设置】,申请js权限。申请理由举例「从网上找的,当时直接复制上就通过了」:尊敬的博客园管理员:您好,我想通过js定制化我的博客,麻烦通过下我的申请。谢谢!通过审核后,......