找回密码
 立即注册
首页 业界区 业界 Next.js项目App目录如何简单集成markdown博客 ...

Next.js项目App目录如何简单集成markdown博客

屠焘 前天 18:15
文章原文:Next.js项目App目录如何简单集成markdown博客
此教程适用于比较简单的项目实现,如果你是刚入门next,并且不想用太复杂的方式去实现一个博客项目,那么这个教程就挺适合你的。
Next.js官方关于markdown的文档有说明过如何渲染markdown,也是针对App目录的,但我尝试过并不太行,可能是版本的问题,不管怎么样,最后我并没有解决这个问题,而是用了别的方案去实现。
此教程适用于app目录的next项目,下面的例子刚好是多语言结构的项目。
实现思路

结合文件结构解说一下大致逻辑:
1.png

Markdown文件放在/app/_articles/[lang]文件夹下管理,如果你是多语言目录,那么每个语种都是单独一个文件夹,如果不是,那么可以直接放在/app/_articles文件夹下。
另外markdown文件里从第一行开始可以放入一些Frontmatter,一般放在文件开头,用---符号分割开,提供一些额外信息,如发布时间、更新时间,是否已经发布,对应的描述,这类的信息可以自定义的,方便你做很多个性化的操作,一般我用来做meta信息的填充。
这里可以给一些Frontmatter的例子:
  1. ---
  2. title: "这是博客标题"
  3. createdAt: "2024-11-12"
  4. updatedAt: "2024-11-12"
  5. isPublished: true
  6. description: "这是博客描述"
  7. ---
复制代码
随着你文件的增多,你需要一些代码来管理、显示你的markdown信息,比如:

  • 在你的blog页面展示所有的markdown博客。
  • 根据markdown文件名称跳转对应的博客详情,比如访问https://i18ncode.com/blog/how-nextjs-app-simply-make-i18n 能正常显示how-nextjs-app-simply-make-i18n.mdx文件内的文本。
  • 渲染markdown文本,当然要包括对应页面的meta信息。
具体代码

大致要做的事情如上所述,下面贴对应的代码。
先封装好一些通用方法在/lib/mdx.ts文件中,方便后续调用:
  1. // mdx.ts
  2. import fs from "fs";
  3. import path from "path";
  4. import matter from "gray-matter";
  5. import readingTime from "reading-time";
  6. const articlesDirectory = path.join(process.cwd(), "app/_articles");
  7. const webContentDirectory = path.join(process.cwd(), "app/_contents");
  8. // 获取 MDX/MD 原始数据
  9. export function getMdxRawData(fileName: string, lang: string, hasSuffix: boolean) {
  10.     let fullPath = path.join(articlesDirectory, lang, `${fileName}`);
  11.     let suffix = hasSuffix // 判断是否有后缀,没有的话就加上后缀
  12.         ? ""
  13.         : fs.existsSync(`${fullPath}.mdx`)
  14.             ? ".mdx"
  15.             : ".md";
  16.     const fileContents = fs.readFileSync(`${fullPath}${suffix}`, "utf8");
  17.     return fileContents;
  18. }
  19. // 处理 MDX/MD 原始数据中的 frontmatter
  20. export function getMdxFrontmatter(mdxRawData: string) {
  21.     const { content, data } = matter(mdxRawData);
  22.     return {
  23.         content,
  24.         frontmatter: data,
  25.         readingTime: readingTime(content).text, // 计算阅读时间
  26.     };
  27. }
  28. // 获取文章的所有信息
  29. export function getArticlesData(fileName: string, lang: string, hasSuffix = false) {
  30.     return {
  31.         ...getMdxFrontmatter(getMdxRawData(fileName, lang, hasSuffix)),
  32.         fileName: fileName.split(".").slice(0, -1).join("."), // 去除后缀
  33.     };
  34. }
  35. // 获取 _articles 目录下的所有文章
  36. export function getAllArticlesData(lang: string) {
  37.     const fileNames = fs.readdirSync(articlesDirectory + "/" + lang);
  38.     const allArticlesData = fileNames.map((fileName) => {
  39.         return getArticlesData(fileName, lang,true);
  40.     });
  41.     return allArticlesData;
  42. }
复制代码
你可以根据你项目的具体情况来调整上面的代码。
在你的blog页面展示所有的markdown博客

调用上面封装好的getAllArticlesData方法,该方法支持一个叫lang的参数,这是多语言项目里有的参数,如果你传入的值为en,那么它就会去/app/_articles/en下获取所有的markdown文件。
然后不要忘记按时间排序:
  1. export default async function BlogPage({params: {lang}}: { params: { lang: Locale } }) {
  2.     const allArticlesData = getAllArticlesData(lang);
  3.     const dictionary = await getDictionary(lang);
  4.     const sortedArticles = allArticlesData.sort((a, b) => {
  5.         // 将日期字符串转换为日期对象
  6.         const dateA = new Date(a.frontmatter.createdAt).getTime();
  7.         const dateB = new Date(b.frontmatter.createdAt).getTime();
  8.         // 比较日期,返回值决定排序
  9.         return dateB - dateA; // 倒序排序
  10.     });
  11.     return (
  12.         
  13.             
  14.                 <h1 className={title()}>{dictionary.blog.title}</h1>
  15.                
  16.                     {sortedArticles.map(article => (
  17.                         <Blog blog={article} key={article.fileName} lang={lang} />
  18.                     ))}
  19.                
  20.             
  21.             <CallToAction dictionary={dictionary} />
  22.         
  23.     );
  24. }
复制代码
根据markdown文件名称跳转对应的博客详情

Blog组件中使用简单的跳转:
  1. [/code]将文件名传递过去,详情页面会根据文件名找到对应的文件进行渲染。
  2. [size=5]渲染markdown文本[/size]
  3. 在/app/[lang]/blog/[id]/page.tsx页面下则是对具体的markdown进行解析和渲染,将对应的内容填入页面,渲染meta信息:
  4. [code]import { getArticlesData } from "@/lib/mdx";
  5. import { Remarkable } from 'remarkable';
  6. import hljs from 'highlight.js';
  7. import {getDictionary} from "@/get-dictionaries";
  8. import CallToAction from "@/components/cta";
  9. import React from "react";
  10. export const generateMetadata = async ({ params }: any) => {
  11.     const { content, frontmatter, readingTime } = getArticlesData(params.id, params.lang);
  12.     const lang = await getDictionary(params.lang);
  13.     return {
  14.         title: frontmatter.title + " | " + lang.blog.meta.title,
  15.         description: frontmatter.description,
  16.         openGraph: {
  17.             title: frontmatter.title + " | " + lang.blog.meta.title,
  18.             type: "website",
  19.             url: ``,
  20.             images: [
  21.                 {
  22.                     // 此处还可以有width和height属性,see:https://medium.com/@moh.mir36/open-graph-with-next-js-v13-app-directory-22c0049e2087
  23.                     url: "/logo.png",
  24.                     alt: ""
  25.                 }
  26.             ],
  27.             siteName: "",
  28.             description: frontmatter.description,
  29.             locale: ""
  30.         },
  31.         twitter: {
  32.             images: [
  33.                 {
  34.                     url: "/logo.png",
  35.                     alt: ""
  36.                 }
  37.             ],
  38.             title: frontmatter.title + " | " + lang.blog.meta.title,
  39.             description: frontmatter.description,
  40.             card: "summary_large_image"
  41.         },
  42.     }
  43. }
  44. // !important:博客的排版需要在tailwind.config.js中添加插件:require("@tailwindcss/typography"),自行查看对应代码
  45. const Page = async ({ params }: any) => {
  46.     const { content, frontmatter, readingTime } = getArticlesData(params.id, params.lang);
  47.     const md = new Remarkable({
  48.         html: true,
  49.         breaks: true,
  50.         linkify: true,
  51.         typographer: true,
  52.         highlight: function (str: string, lang: string) {
  53.             if (lang && hljs.getLanguage(lang)) {
  54.                 try {
  55.                     return hljs.highlight(lang, str).value;
  56.                 } catch (err) {}
  57.             }
  58.             try {
  59.                 return hljs.highlightAuto(str).value;
  60.             } catch (err) {
  61.             }
  62.             return ''; // use external default escaping
  63.         }
  64.     });
  65.     const blog = md.render(content, frontmatter);
  66.     const dictionary = await getDictionary(params.lang);
  67.     return (
  68.         <main className="container pb-24 text-start">
  69.             
  70.                
  71.             
  72.             <CallToAction dictionary={dictionary} />
  73.         </main>
  74.     );
  75. };
  76. export default Page;
复制代码
这里用了Remarkable方案代替了Next的MDXRemote组件。
到这里基本上完成了一半,但是样式方面可能会用欠缺,需要在tailwind.config.js中添加插件:require("@tailwindcss/typography"),代码如下:
  1. import {nextui} from '@nextui-org/theme'
  2. /** @type {import('tailwindcss').Config} */
  3. module.exports = {
  4.     //...
  5.     plugins: [
  6.         // ....
  7.         require("@tailwindcss/typography"), // markdown typography
  8.     ],
  9. }
复制代码
OK,到这里基本大功告成,就可以正常显示了,当然,过程中需要安装一些依赖,根据你项目里缺的依赖来安装就可以了
关于多语言Markdown文件的管理和翻译

你可以看到,使用这种方式,如果是多语言的站点,那么你不可避免地要翻译和管理好对应的markdown文件。
用gpt翻译的话长度会受限制,第一个语种还好,第二个语种之后就会开始忘记原文,然后就开始胡言乱语了;要么你就每次对话都带上原文让gpt翻译,这样对话没几轮就得开启一个新的对话了。
我刚开始做这类工作的时候完成一篇博客需要一整个下午的时间,这实在是太耗时了。
机器翻译更无法接受,它无法识别markdown的符号,会格式错乱,另外机翻效果略显生硬。
基于这块的考虑我做了个专门针对这种情况的翻译器,有需要的朋友可以体验一下markdown翻译器。
markdown翻译器考虑了长度问题,做了文本切割并分段请求,你可以把一整个markdown文本塞进去翻译,直接获取最后的整体结果,经过反复尝试我这是没什么问题的;另外也做了markdown格式的识别和保留,不用害怕丢失格式;最后也考虑了本土化的情况,同样的文本也尽量要求AI用更本土化的方式表达出来,应该是比较适合做国际化的朋友了。
最后,感谢你阅读到这里,博客处会时不时更新一些独立开发的技术分享,希望能为更多的开发者朋友提供一些工具以外的帮助吧。

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
您需要登录后才可以回帖 登录 | 立即注册