gutenbergdocs/docs/getting-started/tutorial.md
2025-10-22 01:40:18 +08:00

42 KiB
Raw Blame History

优化 render.php

最后一步是优化 render.php 文件。如果 currentYear 属性与 fallbackCurrentYear 属性相同,则无需动态生成区块内容。相关内容已保存在数据库中,并可通过 $content 变量在 render.php 文件中调用。

因此,请更新该文件,使其在 currentYearfallbackCurrentYear 不匹配时渲染生成的内容。

$current_year = date( "Y" );

// 确定要显示的内容
if ( isset( $attributes['fallbackCurrentYear'] ) && $attributes['fallbackCurrentYear'] === $current_year ) {

	// 当前年份与备用值相同,直接使用数据库中保存的区块内容(由 save.js 函数生成)
	$block_content = $content;
} else {

	// 当前年份与备用值不同,渲染更新后的区块内容
	if ( ! empty( $attributes['startingYear'] ) && ! empty( $attributes['showStartingYear'] ) ) {
		$display_date = $attributes['startingYear'] . '' . $current_year;
	} else {
		$display_date = $current_year;
	}

	$block_content = '<p ' . get_block_wrapper_attributes() . '>© ' . esc_html( $display_date ) . '</p>';
}

echo wp_kses_post( $block_content );

大功告成!现在您已拥有一个同时支持动态与静态渲染的区块。

收官之笔

恭喜您完成本教程并成功构建专属的版权日期区块!通过这段学习旅程,您已掌握 WordPress 区块开发的基础知识,现在可以开始创建自己的区块了。

本教程完整代码请参考 GitHub 上的 区块开发示例库

若您希望精进技能、挑战更复杂的项目或紧跟 WordPress 最新动态,以下资源将助您提升区块开发能力:

请记住:每位专家都曾是初学者。持续学习、勇于实践,最重要的是,享受使用 WordPress 构建作品的乐趣!

教程:构建你的第一个区块

在本教程中,你将构建一个"版权日期区块"——这是一个基础但实用的区块,用于显示版权符号(©)、当前年份以及可选的起始年份。这类内容通常用于网站页脚。

本教程将引导你完成从使用 create-block 包搭建区块插件脚手架到修改每个文件的完整流程。虽然具备WordPress开发经验会更有帮助但这不是学习本教程的必要条件。

完成本指南后你将清晰理解区块开发的基础知识并掌握创建自定义WordPress区块的必要技能。

你将构建什么

快速预览即将构建的内容。

即将构建的内容

你也可以在 WordPress Playground 中体验完成后的项目,或使用快速入门指南将完整区块插件安装到本地WordPress环境中。

前置要求

完成本教程需要:

  1. 代码编辑器
  2. Node.js开发工具
  3. 本地WordPress环境

若缺少其中任何一项,区块开发环境文档将帮助你完成配置。准备就绪后请返回继续学习。

本教程使用 wp-env 创建本地WordPress开发环境。当然你也可以使用任何满足上述前置要求的开发环境。

搭建区块脚手架

创建版权日期区块的第一步是使用 @wordpress/create-block 包搭建初始区块结构。

建议先阅读create-block入门指南了解该工具包的基本用法。

你可以在计算机的任意目录中运行 create-block,然后使用 wp-env 创建已安装并激活新区块插件的本地WordPress开发环境。

因此请选择存放区块插件的目录,也可新建名为"Block Tutorial"的文件夹。打开终端并进入该目录,然后运行以下命令。

若未使用 wp-env请通过终端进入本地WordPress安装目录的 plugins/ 文件夹再运行以下命令。
npx @wordpress/create-block@latest copyright-date-block --variant=dynamic
cd copyright-date-block

执行命令后,插件文件夹中会出现名为 copyright-date-block 的新目录,其中包含定制区块所需的所有初始文件。

此命令同时建立了区块的基础结构,copyright-date-block 将作为区块在WordPress中的唯一标识符。

你可能注意到命令中使用了 --variant=dynamic 标志,这表示要搭建动态渲染区块的脚手架。后续章节将介绍动态与静态渲染的区别,并为该区块添加静态渲染功能。

前往WordPress后台的插件页面确认插件已激活。接着新建页面或文章确保可以插入版权日期区块。插入后的效果应如下图所示。

编辑器中的区块脚手架

更新 render.php 文件

虽然区块在编辑器中运行正常,但前端仍会显示默认的区块提示信息。要解决此问题,请打开 render.php 文件,您将看到以下内容:

<?php
...
?>
<p <?php echo get_block_wrapper_attributes(); ?>>
	<?php esc_html_e( 'Copyright Date Block  hello from a dynamic block!', 'copyright-date-block' ); ?>
</p>

与编辑器中的 useBlockProps() 函数类似,get_block_wrapper_attributes() 会在区块包装元素中输出所有必要的 CSS 类和样式。只需更新内容部分即可。

您可以使用 date( "Y" ) 在 PHP 中获取当前年份,修改后的 render.php 文件应如下所示:

<?php
...
?>
<p <?php echo get_block_wrapper_attributes(); ?>>© <?php echo date( "Y" ); ?></p>

保存文件后,请确认区块在编辑器和前端均能正确显示。

清理文件

当使用 create-block 工具包搭建区块时,可能会包含一些非必需文件。在本教程中,该区块并未使用样式表或前端 JavaScript。请通过以下操作清理插件的 src/ 目录:

  1. edit.js 文件中移除导入 editor.scss 的代码行
  2. index.js 文件中移除导入 style.scss 的代码行
  3. 删除 editor.scss、style.scss 和 view.js 文件

最后请确保所有修改均已保存,然后终止 npm run start 命令。运行 npm run build 以优化代码并完成生产环境准备。

至此您已构建了一个功能完整的 WordPress 区块,但我们的工作尚未结束。后续章节将继续添加功能并启用静态渲染。

添加区块属性

当前构建的版权日期区块仅显示当前年份,但如果需要同时显示起始年份该如何实现?

即将构建的功能效果

此功能需要用户在区块的某个位置输入起始年份,并且还应具备启用/禁用该功能的开关。

实现方式可以多样化,但都需要使用区块属性。属性允许您为区块存储自定义数据,这些数据随后可用于修改区块的标记。

要实现起始年份功能,您需要定义一个属性存储起始年份,另一个属性用于控制是否显示起始年份。

更新 block.json 文件

区块属性通常在 block.json 文件中定义。请打开该文件,在 example 属性后添加以下配置段:

"example": {},
"attributes": {
	"showStartingYear": {
		"type": "boolean"
	},
	"startingYear": {
		"type": "string"
	}
},

定义属性时必须指明 type 类型。本例中 showStartingYear 需存储布尔值,因此设为 boolean 类型;startingYear 则设为字符串类型。

保存文件后,即可开始编辑器的调整工作。

更新 edit.js 文件

打开 edit.js 文件,您需要完成两项任务:

  • 添加用户界面控件,允许用户输入起始年份、切换功能开关,并将这些设置存储为属性
  • 根据已定义的属性更新区块显示内容

更新 render.php 文件

虽然编辑器看起来很棒,但起始年份功能尚未添加到前端。让我们通过更新 render.php 文件来解决这个问题。

首先,添加一个名为 $display_date 的变量,并复制你在上面的 Edit() 函数中所做的操作。

这个变量应该显示 startingYear 属性和 $current_year 变量的值,用破折号分隔;如果 showStartingYear 属性为 false,则仅显示 $current_year

render.php 中暴露了三个变量,你可以使用它们来自定义块的输出:

  • $attributes(数组):块的属性。
  • $content(字符串):块的默认内容。
  • $blockWP_Block块的实例。

代码应该看起来像这样。

if ( ! empty( $attributes['startingYear'] ) && ! empty( $attributes['showStartingYear'] ) ) {
	$display_date = $attributes['startingYear'] . '' . $current_year;
} else {
	$display_date = $current_year;
}

接下来,你只需要更新块内容,使用 $display_date 而不是 $current_year 变量。

你最终的 render.php 文件应该如下所示。

<?php
$current_year = date( "Y" );

if ( ! empty( $attributes['startingYear'] ) && ! empty( $attributes['showStartingYear'] ) ) {
	$display_date = $attributes['startingYear'] . '' . $current_year;
} else {
	$display_date = $current_year;
}
?>
<p <?php echo get_block_wrapper_attributes(); ?>>
    © <?php echo esc_html( $display_date ); ?>
</p>

保存文件,并确认正确的块内容现在出现在你网站的前端。

你现在已经成功构建了一个动态渲染的自定义块它利用了块支持、WordPress 核心组件和自定义属性。在许多情况下,对于一个显示版权日期并具有一些额外功能的块来说,这已经足够了。

然而,在下一节中,你将向块添加静态渲染。这个练习将说明块数据是如何存储在 WordPress 中的,并在此插件意外禁用时提供回退方案。

添加静态渲染

一个块可以利用动态渲染、静态渲染,或者两者兼用。到目前为止,你构建的块是动态渲染的。它的块标记和关联的属性存储在数据库中,但其 HTML 输出并未存储。

静态渲染的块将始终将块标记、属性和输出存储在数据库中。块还可以在数据库中存储静态输出,同时在前端动态进一步增强,这是两种方法的结合。

如果你在编辑器中切换到代码编辑器,你将看到以下内容。

<!-- wp:create-block/copyright-date-block {"showStartingYear":true,"startingYear":"2017"} /-->

将其与静态渲染的块(如段落块)进行比较。

<!-- wp:paragraph -->
<p>这是一个测试。</p>
<!-- /wp:paragraph -->

段落的 HTML 存储在文章内容中,并保存在数据库中。

你可以在基础文档中了解更多关于动态和静态渲染的内容。虽然大多数块要么是动态渲染,要么是静态渲染,但你可以构建一个同时使用两种方法的块。

为什么要添加静态渲染?

当你向动态渲染的块添加静态渲染时,render.php 文件仍将控制前端的输出,但块的 HTML 内容将被保存在数据库中。这意味着,即使插件从网站中移除,内容仍然会保留。对于这个版权日期块,内容将恢复为一个自定义 HTML 块,你可以轻松地将其转换为段落块。

当块类型不再存在时,编辑器中的错误消息

虽然并非在所有情况下都需要,但向动态渲染的块添加静态渲染可以在插件意外禁用时提供有用的回退方案。

此外,考虑一种情况,块标记包含在块模式或主题模板中。如果用户安装该主题或使用该模式时未安装版权日期块,他们将收到一个块不可用的通知,但内容仍将显示。

添加静态渲染也是探索块内容在 WordPress 中如何存储和渲染的好方法。

文件结构审阅

在开始修改脚手架生成的区块之前,有必要先了解插件的文件结构。请用代码编辑器打开插件文件夹。

构成区块插件的文件

接下来查阅区块文件结构文档,详细了解每个文件的功能。如果现在觉得内容太多不必担心,本教程将逐步讲解每个文件的使用方法。

由于您创建的是动态区块,因此不会看到save.js文件。后续教程中将会把这个文件添加到插件以实现静态渲染,敬请关注。

初始设置

让我们从创建最简单的版权日期区块开始,这将是一个动态渲染的区块,仅显示版权符号(©)和当前年份。我们还将添加若干控件,允许用户修改字体大小和文本颜色。

在执行后续步骤前,请在插件目录的终端中运行 npm run start 命令。该命令将监听 /src 文件夹中每个文件的变更。每次保存文件时,区块的构建文件都会自动更新。

请参阅区块编辑器的JavaScript开发指南文档以了解更多信息。

更新block.json

打开 /src 文件夹中的 block.json 文件。

{
	"$schema": "https://schemas.wp.org/trunk/block.json",
	"apiVersion": 3,
	"name": "create-block/copyright-date-block",
	"version": "0.1.0",
	"title": "Copyright Date Block",
	"category": "widgets",
	"icon": "smiley",
	"description": "Example block scaffolded with Create Block tool.",
	"example": {},
	"supports": {
		"html": false
	},
	"textdomain": "copyright-date-block",
	"editorScript": "file:./index.js",
	"editorStyle": "file:./index.css",
	"style": "file:./style-index.css",
	"render": "file:./render.php",
	"viewScript": "file:./view.js"
}
建议查阅block.json文档了解此文件的基本信息。

虽然脚手架工具已生成此文件,但需要根据版权日期区块的需求进行一些调整。

修改区块标识

首先移除图标并添加更合适的描述(后续将添加自定义图标):

  1. 删除 icon 这一行
  2. 将描述更新为“显示您网站的版权日期”
  3. 保存文件

刷新编辑器后,您会看到区块不再显示笑脸图标,且描述已更新。

更新信息后的编辑器区块显示

添加区块支持功能

接下来添加若干区块支持功能,以便用户控制区块的字体大小和文本颜色。

在构建自定义功能前,应优先使用原生区块支持功能。这种方式能为用户提供跨区块的一致编辑体验,且仅需少量代码即可让区块获得核心功能支持。

block.json 文件中的 supports 部分更新为:

"supports": {
	"color": {
		"background": false,
		"text": true
	},
	"html": false,
	"typography": {
		"fontSize": true
	}
},

请注意当通过 "text": true 启用文本颜色支持时,背景颜色功能默认也会启用。您可以保留此设置,但本教程不需要该功能,因此可手动设置为 "background": false

保存文件后在编辑器中选择区块,您将在设置面板中看到颜色和版式面板。尝试修改这些设置并观察效果。

具备区块支持功能的编辑器区块显示

添加保存功能

首先在 src/ 文件夹中新建名为 save.js 的文件,并添加以下内容:

import { useBlockProps } from '@wordpress/block-editor';

export default function save() {
	return (
		<p { ...useBlockProps.save() }>
			{ 'Copyright Date Block  hello from the saved content!' }
		</p>
	);
}

这段代码与原有的 edit.js 文件结构相似,更多细节可参考区块包装器文档。

接着在 index.js 文件中导入这个 save() 函数,并为 registerBlockType() 函数添加 save 属性。以下是更新后文件的简化视图:

import save from './save';

...

registerBlockType( metadata.name, {
	icon: calendarIcon,
	edit: Edit,
	save
} );
在定义对象属性时,若属性名与变量名相同,可使用属性简写语法。这就是为什么上面的代码使用 save 而非 save: save

保存 save.jsindex.js 文件后刷新编辑器,你将看到如下界面:

编辑器中显示的区块验证错误信息

不必担心,这个错误是预期内的。打开浏览器检查器,你会看到以下信息:

控制台显示的区块验证错误信息

出现区块验证错误是因为 save() 函数返回了区块内容但由于之前保存的是动态区块区块标记中并未存储HTML。当前标记结构如下所示

<!-- wp:create-block/copyright-date-block {"showStartingYear":true,"startingYear":"2017"} /-->

在后续步骤中更新 save() 函数时,你会多次遇到这类错误。只需点击“尝试恢复区块”并刷新页面即可。

完成区块恢复后,打开代码编辑器会看到标记已更新为:

<!-- wp:create-block/copyright-date-block {"showStartingYear":true,"startingYear":"2017"} -->
<p class="wp-block-create-block-copyright-date-block">Copyright Date Block  hello from the saved content!</p>
<!-- /wp:create-block/copyright-date-block -->

使用静态渲染构建区块时经常会遇到验证错误,这属于正常现象。save() 函数的输出必须与文章内容中的HTML完全匹配随着功能添加可能会出现不同步。只要在完整构建区块后没有验证错误就表示大功告成。

更新 save.js

接下来更新 save() 函数的输出以显示正确内容。首先采用与 edit.js 相同的实现方式:

  1. 为函数添加 attributes 参数
  2. 定义 showStartingYearstartingYear 变量
  3. 定义 currentYear 变量
  4. 根据 currentYearshowStartingYearstartingYear 的值定义 displayDate 变量

最终代码应如下所示:

export default function save( { attributes } ) {
	const { showStartingYear, startingYear } = attributes;
	const currentYear = new Date().getFullYear().toString();

	let displayDate;

	if ( showStartingYear && startingYear ) {
		displayDate = startingYear + '' + currentYear;
	} else {
		displayDate = currentYear;
	}

	return (
		<p { ...useBlockProps.save() }>© { displayDate }</p>
	);
}

保存文件并刷新编辑器,点击“尝试恢复区块”后更新页面。在代码编辑器中查看,区块标记现在应显示为:

<!-- wp:create-block/copyright-date-block {"showStartingYear":true,"startingYear":"2017"} -->
<p class="wp-block-create-block-copyright-date-block">© 20172023</p>
<!-- /wp:create-block/copyright-date-block -->

至此看似已完成所有工作——区块内容已以HTML形式存入数据库前端输出也能动态渲染。但仍有几个问题需要解决。

试想这种情况用户在2023年将区块添加到页面然后在2024年重新编辑该页面。前端显示会正常更新但在编辑器中会出现区块验证错误。因为 save() 函数识别当前是2024年而数据库中存储的区块内容仍显示2023年。

我们将在下一节解决这个问题。

切换控件

接下来,我们添加一个用于开启或关闭起始年份功能的切换开关。您可以通过设置 showStartingYear 属性的 ToggleControl 组件实现。该组件应包含:

  1. 设置为“显示起始年份”的 label 属性
  2. 设置为 showStartingYear 属性的 checked 属性
  3. 在切换开关被勾选或取消时更新 showStartingYear 属性的 onChange 属性

您还可以更新“起始年份”文本输入框,使其仅在 showStartingYeartrue 时显示,这可以通过 && 逻辑运算符实现。

Edit() 函数应如下所示。

export default function Edit( { attributes, setAttributes } ) {
	const { showStartingYear, startingYear } = attributes;
	const currentYear = new Date().getFullYear().toString();

	return (
		<>
			<InspectorControls>
				<PanelBody title={ __( '设置', 'copyright-date-block' ) }>
					<ToggleControl
						checked={ !! showStartingYear }
						label={ __(
							'显示起始年份',
							'copyright-date-block'
						) }
						onChange={ () =>
							setAttributes( {
								showStartingYear: ! showStartingYear,
							} )
						}
					/>
					{ showStartingYear && (
						<TextControl
							__nextHasNoMarginBottom
							__next40pxDefaultSize
							label={ __(
								'起始年份',
								'copyright-date-block'
							) }
							value={ startingYear || '' }
							onChange={ ( value ) =>
								setAttributes( { startingYear: value } )
							}
						/>
					) }
				</PanelBody>
			</InspectorControls>
			<p { ...useBlockProps() }>© { currentYear }</p>
		</>
	);
}

保存文件并刷新编辑器。确认点击切换开关会显示文本输入框,并且在更新页面时切换开关保持激活状态。

在设置侧边栏中编辑新的“显示起始年份”切换开关的实时预览

更新区块内容

到目前为止,您已经创建了用于添加起始年份并更新相关区块属性的用户界面。现在需要实际更新编辑器中的区块内容。

我们创建一个名为 displayDate 的新变量。当 showStartingYeartrue 且用户提供了 startingYear 时,displayDate 应包含由破折号连接的 startingYearcurrentYear。否则,仅显示 currentYear

代码应如下所示。

let displayDate;

if ( showStartingYear && startingYear ) {
	displayDate = startingYear + '' + currentYear;
} else {
	displayDate = currentYear;
}
当您使用 let 声明变量时,表示该变量后续可能会被重新赋值。使用 const 声明变量则表示该变量永远不会改变。您也可以使用 const 重写此代码,这仅是个人偏好问题。

接下来,您只需更新区块内容,使用 displayDate 替代 currentYear 变量。

Edit() 函数应如下所示。

export default function Edit( { attributes, setAttributes } ) {
	const { showStartingYear, startingYear } = attributes;
	const currentYear = new Date().getFullYear().toString();

	let displayDate;

	if ( showStartingYear && startingYear ) {
			displayDate = startingYear + '' + currentYear;
	} else {
		displayDate = currentYear;
	}

	return (
		<>
			<InspectorControls>
				<PanelBody title={ __( '设置', 'copyright-date-block' ) }>
					<ToggleControl
						checked={ !! showStartingYear }
						label={ __(
							'显示起始年份',
							'copyright-date-block'
						) }
						onChange={ () =>
							setAttributes( {
								showStartingYear: ! showStartingYear,
							} )
						}
					/>
					{ showStartingYear && (
						<TextControl
							label={ __(
								'起始年份',
								'copyright-date-block'
							) }
							value={ startingYear || '' }
							onChange={ ( value ) =>
								setAttributes( { startingYear: value } )
							}
						/>
					) }
				</PanelBody>
			</InspectorControls>
			<p { ...useBlockProps() }>© { displayDate }</p>
		</>
	);
}

保存文件并刷新编辑器。确认当您在设置面板中进行更改时,区块内容会正确更新。

设置侧边栏中新字段更新区块内容的实时预览

处理静态渲染区块中的动态内容

通常来说,您需要避免在静态渲染区块中使用动态内容。这也是为什么在提及动态渲染时会使用"动态"这个术语的原因之一。

不过在本教程中,您将结合使用两种渲染方法,只需要添加少量代码就能在年份变更时避免区块验证错误。

问题的根源在于currentYear变量是在save()函数中动态设置的。实际上,这应该是函数内的静态变量,可以通过添加属性来解决这个问题。

添加新属性

打开block.json文件,添加一个名为fallbackCurrentYear、类型为string的新属性。此时文件的attributes部分应如下所示:

"attributes": {
    "fallbackCurrentYear": {
		"type": "string"
    },
    "showStartingYear": {
		"type": "boolean"
    },
    "startingYear": {
		"type": "string"
	}
},

接下来打开save.js文件,使用新的fallbackCurrentYear属性替代currentYear。更新后的save()函数应如下所示:

export default function save( { attributes } ) {
	const { fallbackCurrentYear, showStartingYear, startingYear } = attributes;

	let displayDate;

	if ( showStartingYear && startingYear ) {
		displayDate = startingYear + '' + fallbackCurrentYear;
	} else {
		displayDate = fallbackCurrentYear;
	}

	return (
		<p { ...useBlockProps.save() }>© { displayDate }</p>
	);
}

现在,如果fallbackCurrentYear未定义会发生什么?

之前currentYear是在函数内定义的,因此即使showStartingYearstartingYear未定义,save()函数也始终有内容可以返回。

与其只返回版权符号,不如添加一个条件判断:如果fallbackCurrentYear未设置,则返回null。通常来说,在数据库中保存空内容比保存不完整的数据更好。

最终的save()函数应如下所示:

export default function save( { attributes } ) {
	const { fallbackCurrentYear, showStartingYear, startingYear } = attributes;

	if ( ! fallbackCurrentYear ) {
		return null;
	}

	let displayDate;

	if ( showStartingYear && startingYear ) {
		displayDate = startingYear + '' + fallbackCurrentYear;
	} else {
		displayDate = fallbackCurrentYear;
	}

	return (
		<p { ...useBlockProps.save() }>© { displayDate }</p>
	);
}

保存block.jsonsave.js文件,您无需再进行其他修改。

在edit.js中设置属性

既然save()函数现在使用新的fallbackCurrentYear,就需要在某个地方设置这个值。让我们使用Edit()函数来实现。

打开edit.js文件,首先在Edit()函数顶部与其他属性一起定义fallbackCurrentYear变量。接着检查函数中的现有逻辑。

当区块在编辑器中加载时,会定义currentYear变量。函数随后使用该变量来设置区块内容。

现在,让我们在区块加载时,如果fallbackCurrentYear属性尚未设置,就将其设置为currentYear的值:

if ( currentYear !== fallbackCurrentYear ) {
	setAttributes( { fallbackCurrentYear: currentYear } );
}

这样虽然可行但可以通过确保此代码仅在区块初始化时运行一次来进行优化。为此您可以使用React的useEffect钩子。有关如何使用此钩子的更多信息请参阅React文档。

首先通过以下代码导入useEffect

import { useEffect } from 'react';

然后将上面的setAttribute()代码包装在useEffect中,并将这段代码放在Edit()函数的currentYear定义之后。最终结果应如下所示:

export default function Edit( { attributes, setAttributes } ) {
	const { fallbackCurrentYear, showStartingYear, startingYear } = attributes;

	// 获取当前年份并确保其为字符串类型
	const currentYear = new Date().getFullYear().toString();

	// 当区块加载时如果fallbackCurrentYear尚未设置
	// 则将其属性设置为当前年份
	useEffect( () => {
		if ( currentYear !== fallbackCurrentYear ) {
			setAttributes( { fallbackCurrentYear: currentYear } );
		}
	}, [ currentYear, fallbackCurrentYear, setAttributes ] );

	...

当区块在编辑器中初始化时,fallbackCurrentYear属性将立即被设置。这个值随后可供save()函数使用,正确的区块内容将显示且不会出现区块验证错误。

唯一需要注意的是年份变更的情况。如果版权日期区块在2023年添加到页面然后在2024年进行编辑fallbackCurrentYear属性将不再等于currentYear,该属性将自动更新为2024。这将更新save()函数返回的HTML。

虽然您不会遇到任何区块验证错误,但编辑器会检测到页面已发生更改,并提示您进行更新。

添加用户界面

在本教程的前面部分,您添加了区块支持功能,这些功能会在区块的设置侧边栏中自动创建颜色和排版面板。您可以使用 InspectorControls 组件创建自定义面板。

检查器控件

InspectorControls 属于 @wordpress/block-editor 包,因此您可以通过在第 14 行添加组件名称将其导入到 edit.js 文件中。结果应如下所示:

import { InspectorControls, useBlockProps } from '@wordpress/block-editor';

接下来,更新 Edit 函数,使其返回当前区块内容和一个包含“测试”文本的 InspectorControls 组件。您可以将所有内容包装在 Fragment (<></>) 中,以确保 JSX 语法正确。结果应如下所示:

export default function Edit() {
	const currentYear = new Date().getFullYear().toString();

	return (
		<>
			<InspectorControls>
				测试
			</InspectorControls>
			<p { ...useBlockProps() }>© { currentYear }</p>
		</>
	);
}

保存文件并刷新编辑器。选择区块时,您应在设置侧边栏中看到“测试”消息。

设置侧边栏现在显示消息

组件和面板

现在,让我们使用更多核心组件来添加自定义面板和起始年份功能的用户界面。您需要从 @wordpress/components 包中导入 PanelBodyTextControlToggleControl

edit.js 文件的其他导入下方添加以下行:

import { PanelBody, TextControl, ToggleControl } from '@wordpress/components';

然后将“测试”消息包装在 PanelBody 组件中,并将 title 参数设置为“设置”。有关其他参数选项,请参阅组件文档

export default function Edit() {
	const currentYear = new Date().getFullYear().toString();

	return (
		<>
			<InspectorControls>
				<PanelBody title={ __( '设置', 'copyright-date-block' ) }>
					测试
				</PanelBody>
			</InspectorControls>
			<p { ...useBlockProps() }>© { currentYear }</p>
		</>
	);
}

保存文件并刷新编辑器。您现在应该可以看到新的设置面板。

设置侧边栏现在显示自定义面板

文本控件

下一步是将“测试”消息替换为 TextControl 组件,该组件允许用户设置 startingYear 属性。但在执行此操作之前,您必须在 Edit() 函数中包含两个参数:

  • attributes 是一个包含区块所有属性的对象。
  • setAttributes 是一个允许您更新属性值的函数。

包含这些参数后,您可以获取 showStartingYearstartingYear 属性。

更新 Edit() 函数的开头部分,使其如下所示:

export default function Edit( { attributes, setAttributes } ) {
	const { showStartingYear, startingYear } = attributes;
	...
要查看与版权日期区块关联的所有属性,请在 Edit() 函数的开头添加 console.log( attributes );。这在构建和测试自定义区块时非常有用。

现在,您可以删除“测试”消息并添加 TextControl。它应包括:

  1. 一个设置为“起始年份”的 label 属性。
  2. 一个设置为属性 startingYearvalue 属性。
  3. 一个 onChange 属性,用于在值更改时更新 startingYear 属性。

将所有内容整合后,Edit() 函数应如下所示:

export default function Edit( { attributes, setAttributes } ) {
	const { showStartingYear, startingYear } = attributes;
	const currentYear = new Date().getFullYear().toString();

	return (
		<>
			<InspectorControls>
				<PanelBody title={ __( '设置', 'copyright-date-block' ) }>
					<TextControl
					    __nextHasNoMarginBottom
						__next40pxDefaultSize
						label={ __(
							'起始年份',
							'copyright-date-block'
						) }
						value={ startingYear || '' }
						onChange={ ( value ) =>
							setAttributes( { startingYear: value } )
						}
					/>
				</PanelBody>
			</InspectorControls>
			<p { ...useBlockProps() }>© { currentYear }</p>
		</>
	);
}
您可能已经注意到 value 属性的值为 startingYear || ''。符号 || 称为 逻辑或(逻辑析取)运算符。这可以防止当 startingYear 为空时在 React 中出现警告。详情请参阅 受控和非受控组件

保存文件并刷新编辑器。确认设置面板中现在存在一个文本字段。添加一个起始年份,并确认在更新页面时该值已保存。

在设置侧边栏中编辑新的起始年份字段的实时预览

移除不必要的代码

为简化操作版权日期区块的样式将完全由颜色与排版区块支持功能控制。该区块也不包含任何前端JavaScript代码。因此您无需在block.json文件中指定样式表或viewScript参数。

  1. 删除editorStyle配置行
  2. 删除style配置行
  3. 删除viewScript配置行
  4. 保存文件

刷新编辑器后,您将看到区块样式已与当前主题保持一致。

编辑器中去掉默认样式的区块效果

完整配置汇总

最终完成的block.json文件应如下所示:

{
	"$schema": "https://schemas.wp.org/trunk/block.json",
	"apiVersion": 3,
	"name": "create-block/copyright-date-block",
	"version": "0.1.0",
	"title": "版权日期区块",
	"category": "widgets",
	"description": "显示站点版权日期",
	"example": {},
	"supports": {
		"color": {
			"background": false,
			"text": true
		},
		"html": false,
		"typography": {
			"fontSize": true
		}
	},
	"textdomain": "copyright-date-block",
	"editorScript": "file:./index.js",
	"render": "file:./render.php"
}

更新index.js文件

在开始构建区块功能之前,我们先进行清理工作并为区块添加自定义图标。

打开index.js文件这是区块的主JavaScript文件用于在客户端注册区块。您可以通过区块注册文档深入了解客户端与服务端注册机制。

首先查看registerBlockType函数。该函数接收从导入的block.json文件中获取的区块名称,以及区块配置对象。

import Edit from './edit';
import metadata from './block.json';

registerBlockType( metadata.name, {
	edit: Edit,
} );

默认配置对象仅包含edit属性,但您还可以添加包括icon在内的更多属性。虽然大部分属性已在block.json中定义但若需使用自定义SVG图标则需要在此处专门指定。

添加自定义图标

Gutenberg组件库获取日历图标按如下方式将SVG代码添加到函数中

const calendarIcon = (
	<svg
		viewBox="0 0 24 24"
		xmlns="http://www.w3.org/2000/svg"
		aria-hidden="true"
		focusable="false"
	>
		<path d="M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm.5 16c0 .3-.2.5-.5.5H5c-.3 0-.5-.2-.5-.5V7h15v12zM9 10H7v2h2v-2zm0 4H7v2h2v-2zm4-4h-2v2h2v-2zm4 0h-2v2h2v-2zm-4 4h-2v2h2v-2zm4 0h-2v2h2v-2z"></path>
	</svg>
);

registerBlockType( metadata.name, {
	icon: calendarIcon,
	edit: Edit
} );
所有区块图标尺寸应为24x24像素。请注意上方代码中的viewBox参数。

保存index.js文件并刷新编辑器,您将看到默认图标已替换为日历图标。

编辑器中使用自定义图标的区块

至此,区块的图标与描述信息已正确配置,区块支持功能也允许您调整字体大小和文本颜色。接下来将进入区块实际功能的开发阶段。

更新edit.js文件

edit.js文件控制着区块在编辑器中的功能与显示效果。当前用户会看到“Copyright Date Block hello from the editor!”提示信息,现在我们来修改这一内容。

打开文件可见Edit()函数返回包含默认信息的段落标签:

export default function Edit() {
	return (
		<p { ...useBlockProps() }>
			{ __(
				'Copyright Date Block  hello from the editor!',
				'copyright-date-block-demo'
			) }
		</p>
	);
}

其实际逻辑比表面看起来更简洁:

  • useBlockProps()用于输出编辑器所需的区块包装器中的所有CSS类与样式包含先前添加的区块支持功能提供的样式
  • __()用于实现文本字符串的国际化
建议查阅区块包装器文档,了解如何确保区块标记包装器具备正确属性的入门指南。

需要明确的是,本区块的核心功能是显示版权符号(©)与当前年份。首先需要以字符串格式获取当前年份,可通过以下代码实现:

const currentYear = new Date().getFullYear().toString();

接着更新函数以显示正确信息:

export default function Edit() {
	const currentYear = new Date().getFullYear().toString();

	return (
		<p { ...useBlockProps() }>© { currentYear }</p>
	);
}

保存edit.js文件并刷新编辑器,您现在将看到版权符号(©)后接当前年份的显示效果。

编辑器中正确显示内容的区块