Pagic

_layout.tsx

从这一章开始的内容是提供给需要深度定制的用户的,如果你只是简单使用 Pagic,那么可以直接跳到博客章节。

_layout.tsx 是 Pagic 的核心理念之一。

什么是 _layout.tsx§

_layout.tsx 可以理解为 Pagic 在渲染页面时的模版文件,所有页面文件(md/tsx)在渲染时都会以 _layout.tsx 为模版。

我们不妨在之前的 site 项目中创建一个 _layout.tsx

site/
├── _layout.tsx├── pagic.config.ts
└── README.md

其中 _layout.tsx 的内容如下:

import { React, PagicLayout } from 'https://deno.land/x/pagic@v1.6.3/mod.ts';

const Layout: PagicLayout = ({ title, content }) => (
  <html>
    <head>
      <title>{title}</title>
      <meta charSet="utf-8" />
    </head>
    <body>
      {content}
      <p>Custom _layout.tsx</p>
    </body>
  </html>
);

export default Layout;

接下来我们运行:

pagic build --serve

然后打开 http://127.0.0.1:8000/ ,可以看到页面中除了有标题 Hello world 之外,还有一个段落 Custom _layout.tsx,这说明此页面是用 _layout.tsx 作为模版渲染出来的。

为什么在上一章中不需要 _layout.tsx 也可以构建出页面呢?

那是因为 Pagic 默认会使用 default 主题中的 _layout.tsx 文件作为模版。当我们创建一个自己的 _layout.tsx 时,就会覆盖掉主题中的 _layout.tsx 了。

子页面和子模版§

_layout.tsx 的设计是符合直觉的,当我们创建子目录时,其中的页面会优先使用该目录下的 _layout.tsx,只有当子目录下没有 _layout.tsx 时才会向上级目录查找,直到找到 _layout.tsx 为止:

site/
|── dist    # 构建结果目录
|   |── index.html
|   └── foo
|       ├── index.html
|       └── bar
|           └── index.html
├── _layout.tsx├── pagic.config.ts
|── README.md
└── foo
    ├── README.md
    └── bar
        ├── _layout.tsx        └── README.md

在上面的例子中,site/foo/bar/README.md 会使用同级目录下的 site/foo/bar/_layout.tsx 作为模版,而 site/foo/README.md 则会使用 site/_layout.tsx 作为模版。

通过配置页面头信息可以跳过此规则,强制指定一个模版。

组件化§

组件化是 React 的重要特性之一,我们可以通过拆分 _layout.tsx 为一个个子组件来复用代码。不过在 Pagic 中,由于需要支持 tsx 文件渲染为页面,所以我们需要对子组件做一个约定——以 _ 开头的组件为子组件:

site/
|── dist    # 构建结果目录
|   └── hello.html
├── _layout.tsx
├── _sidebar.tsx├── hello.tsx
└── pagic.config.ts

在上面的例子中,hello.tsx 会被构建为 dist/hello.html,而 _sidebar.tsx 由于是 _ 开头,所以不会被构建为页面。这样就可以实现对 _layout.tsx 的拆分,将 Sidebar 组件拆分到 _sidebar.tsx 文件中,然后在 _layout.tsx 中引用即可:

import { React, PagicLayout } from 'https://deno.land/x/pagic@v1.6.3/mod.ts';

import Sidebar from './_sidebar.tsx';
const Layout: PagicLayout = ({ title, content }) => (
  <html>
    <head>
      <title>{title}</title>
      <meta charSet="utf-8" />
    </head>
    <body>
      <Sidebar />      {content}
    </body>
  </html>
);

export default Layout;

props§

注意到上面的例子中我们取用了 props 中的 titlecontent,那么除了这两个之外 props 中还有哪些属性可以使用呢?

请参考下面的表格:

属性 类型 描述
title string 页面的标题,一般会放到 <head><title>
content ReactElement 页面的内容,一般会放到 <body>
contentTitle ReactElement content 中的标题,和 contentBody 配合使用可以在标题和正文之间插入内容
contentBody ReactElement content 中的正文,和 contentTitle 配合使用可以在标题和正文之间插入内容
toc ReactElement 页面的目录(Table of Content)
author string 该文件的第一个提交者
contributors string[] 该文件的所有提交者(包括第一个提交者),以第一次提交的时间排序(先提交的排在前面)
date Date 该文件第一次提交时的日期
updated Date 该文件最后一次提交的日期
excerpt string 文章的摘要,默认为文章的前 210 个字符
cover string 文章的头图(第一张图片)
tags string[] 文章的标签
categories string[] 文章的分类
config PagicConfig Pagic 运行时[1]的配置
pagePath string 页面路径,如 docs/README.md
layoutPath string 页面的模版路径,如 docs/_layout.tsx
outputPath string 页面的输出路径,如 docs/index.html
head ReactElement 需要插入到 <head> 标签中的内容
script ReactElement script 插件生成的 ReactElement
loading boolean 页面是否在加载中
sidebar PagePropsSidebar sidebar 插件解析后的对象
prev PagePropsSidebar[0] 上一页的详细信息
next PagePropsSidebar[0] 下一页的详细信息
gitalk ReactElement gitalk 插件生成的 ReactElement
blog 博客章节 当前页面的博客信息
language 国际化章节 当前页面的语言
其他 any 第三方插件也可能扩充 props

静态资源§

除了以上提到的特殊文件之外的其他文件均会被视为静态资源,直接复制到 dist 目录下。

现将所有文件名约定汇总如下:

文件名 描述
. 开头 隐藏文件,会被忽略
pagic.config.tspagic.config.tsx 配置文件
_layout.tsx 模版文件
_ 开头的 tsx 文件 子组件
mdtsx 后缀的文件 页面文件
其他文件 静态资源,会被直接复制到 dist 目录下

注解§

  1. Pagic 运行时的配置与 pagic.config.ts 中的配置会有少许差异