在Typecho中实现AI打字机文章摘要功能:从0到1的完整教程

文章摘要
收起

前言

本文将教你如何在Typecho博客中实现一个带有打字机动画效果的文章摘要功能,并且支持展开收起。我们将使用Typecho的自定义字段功能来存储摘要内容,并通过JavaScript为其添加打字机效果。

1. 了解Typecho的自定义字段

Typecho允许我们为每篇文章添加自定义字段。在编辑文章页面的底部,有一个"自定义字段"区域,我们可以在这里添加名为"summary"的字段,用于存储文章摘要。

2. 基础版:在文章页面显示自定义字段内容

首先,我们需要修改文章模板,在适当位置添加代码来显示summary字段的内容。找到你的文章页面模板(通常是post.php),在文章内容区域的开始位置添加以下代码:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded">
        <h5 class="fw-bold">文章摘要</h5>
        <div class="summary-content">
            <?php echo $this->fields->summary; ?>
        </div>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

这段代码会检查文章是否有summary自定义字段,如果有则显示出来。

3. 添加蓝色主题样式

接下来,我们给摘要区域添加蓝色主题样式:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5;">
        <h5 class="fw-bold text-primary">文章摘要</h5>
        <div class="summary-content text-primary" style="color: #2171d3; font-weight: normal;">
            <?php echo $this->fields->summary; ?>
        </div>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

这样,摘要区域就有了蓝色背景和蓝色文字。

4. 添加打字机效果

现在,让我们添加打字机效果,使摘要内容逐字显示:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5;">
        <h5 class="fw-bold text-primary">文章摘要</h5>
        <div class="summary-content text-primary" id="typewriter-text" style="color: #2171d3; font-weight: normal;"></div>
        <script>
        document.addEventListener('DOMContentLoaded', function() {
            // 存储原始内容
            const originalText = `<?php echo addslashes($this->fields->summary); ?>`;
            const typewriterElement = document.getElementById('typewriter-text');
            let i = 0;
            const speed = 30; // 打字速度,数值越小越快
            
            // 打字机效果函数
            function typeWriter() {
                if (i < originalText.length) {
                    typewriterElement.innerHTML += originalText.charAt(i);
                    i++;
                    setTimeout(typeWriter, speed);
                }
            }
            
            // 启动打字机效果
            typeWriter();
        });
        </script>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

这段代码通过JavaScript实现了打字机效果,摘要内容会逐字显示。

5. 添加光标闪烁效果

为了增强打字机效果,我们添加一个闪烁的光标:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5;">
        <h5 class="fw-bold text-primary">文章摘要</h5>
        <div class="summary-content text-primary" id="typewriter-text" style="color: #2171d3; font-weight: normal;"></div>
        <style>
            .cursor {
                display: inline-block;
                width: 2px;
                height: 1em;
                background-color: #2171d3;
                margin-left: 2px;
                animation: blink 0.7s infinite;
                vertical-align: text-bottom;
            }
            
            @keyframes blink {
                0%, 100% { opacity: 1; }
                50% { opacity: 0; }
            }
        </style>
        <script>
        document.addEventListener('DOMContentLoaded', function() {
            // 存储原始内容
            const originalText = `<?php echo addslashes($this->fields->summary); ?>`;
            const typewriterElement = document.getElementById('typewriter-text');
            let i = 0;
            const speed = 30; // 打字速度,数值越小越快
            
            // 创建光标元素
            const cursor = document.createElement('span');
            cursor.className = 'cursor';
            typewriterElement.appendChild(cursor);
            
            // 打字机效果函数
            function typeWriter() {
                if (i < originalText.length) {
                    // 在光标前插入文本
                    const textNode = document.createTextNode(originalText.charAt(i));
                    typewriterElement.insertBefore(textNode, cursor);
                    i++;
                    setTimeout(typeWriter, speed);
                }
            }
            
            // 启动打字机效果
            typeWriter();
        });
        </script>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

现在,摘要内容会逐字显示,每个字符后面都有一个闪烁的蓝色光标。

6. 解决打字机效果引起的页面跳动问题

为了防止打字机效果导致页面内容不断下移,我们需要预先设置容器高度:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5;">
        <h5 class="fw-bold text-primary">文章摘要</h5>
        <!-- 隐藏的div用于计算高度 -->
        <div id="hidden-summary" style="visibility: hidden; position: absolute; width: calc(100% - 3rem); color: #2171d3; padding: 0.5rem 0;"><?php echo $this->fields->summary; ?></div>
        <!-- 实际显示的div -->
        <div class="summary-content text-primary" id="typewriter-text" style="color: #2171d3; font-weight: normal; min-height: 1.5em; padding: 0.5rem 0;"></div>
        <style>
            .cursor {
                display: inline-block;
                width: 2px;
                height: 1em;
                background-color: #2171d3;
                margin-left: 2px;
                animation: blink 0.7s infinite;
                vertical-align: text-bottom;
            }
            
            @keyframes blink {
                0%, 100% { opacity: 1; }
                50% { opacity: 0; }
            }
        </style>
        <script>
        document.addEventListener('DOMContentLoaded', function() {
            // 存储原始内容
            const originalText = `<?php echo addslashes($this->fields->summary); ?>`;
            const typewriterElement = document.getElementById('typewriter-text');
            const hiddenElement = document.getElementById('hidden-summary');
            
            // 根据隐藏元素设置打字机容器高度
            setTimeout(() => {
                const height = hiddenElement.offsetHeight;
                typewriterElement.style.height = height + 'px';
                // 高度设置完成后可以隐藏计算元素
                hiddenElement.remove();
            }, 0);
            
            let i = 0;
            const speed = 30; // 打字速度,数值越小越快
            
            // 创建光标元素
            const cursor = document.createElement('span');
            cursor.className = 'cursor';
            typewriterElement.appendChild(cursor);
            
            // 打字机效果函数
            function typeWriter() {
                if (i < originalText.length) {
                    // 在光标前插入文本
                    const textNode = document.createTextNode(originalText.charAt(i));
                    typewriterElement.insertBefore(textNode, cursor);
                    i++;
                    setTimeout(typeWriter, speed);
                }
            }
            
            // 启动打字机效果
            typeWriter();
        });
        </script>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

通过创建一个隐藏的元素来预先计算高度,我们解决了打字机效果导致的页面跳动问题。

7. 添加展开/收起功能

现在,我们添加一个展开/收起按钮,让用户可以控制摘要的显示:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5;">
        <div class="d-flex justify-content-between align-items-center mb-2">
            <h5 class="fw-bold text-primary mb-0">文章摘要</h5>
            <span id="toggle-summary" class="text-primary" style="cursor: pointer; font-size: 0.9rem;">收起</span>
        </div>
        <!-- 隐藏的div用于计算高度 -->
        <div id="hidden-summary" style="visibility: hidden; position: absolute; width: calc(100% - 3rem); color: #2171d3; padding: 0.5rem 0;"><?php echo $this->fields->summary; ?></div>
        <!-- 实际显示的div -->
        <div class="summary-container" id="summary-container">
            <div class="summary-content text-primary" id="typewriter-text" style="color: #2171d3; font-weight: normal; min-height: 1.5em; padding: 0.5rem 0;"></div>
        </div>
        
        <style>
            .cursor {
                display: inline-block;
                width: 2px;
                height: 1em;
                background-color: #2171d3;
                margin-left: 2px;
                animation: blink 0.7s infinite;
                vertical-align: text-bottom;
            }
            
            @keyframes blink {
                0%, 100% { opacity: 1; }
                50% { opacity: 0; }
            }
            
            .summary-container.collapsed {
                height: 2em; /* 增加高度确保显示完整一行 */
                overflow: hidden;
                margin: 0;
            }
            
            /* 当动画完成时隐藏光标 */
            .cursor.hidden {
                display: none;
            }
        </style>
        
        <script>
        document.addEventListener('DOMContentLoaded', function() {
            // 存储原始内容
            const originalText = `<?php echo addslashes($this->fields->summary); ?>`;
            const typewriterElement = document.getElementById('typewriter-text');
            const hiddenElement = document.getElementById('hidden-summary');
            const summaryContainer = document.getElementById('summary-container');
            const toggleButton = document.getElementById('toggle-summary');
            
            // 展开/收起状态
            let isExpanded = true;
            
            // 根据隐藏元素设置打字机容器高度
            setTimeout(() => {
                const height = hiddenElement.offsetHeight;
                // 增加额外高度容纳底部信息
                typewriterElement.style.height = (height + 10) + 'px';
                // 高度设置完成后可以隐藏计算元素
                hiddenElement.remove();
            }, 0);
            
            // 展开/收起按钮事件
            toggleButton.addEventListener('click', function() {
                isExpanded = !isExpanded;
                if (isExpanded) {
                    summaryContainer.classList.remove('collapsed');
                    toggleButton.textContent = '收起';
                } else {
                    summaryContainer.classList.add('collapsed');
                    toggleButton.textContent = '展开';
                }
            });
            
            let i = 0;
            const speed = 30; // 打字速度,数值越小越快
            
            // 创建光标元素
            const cursor = document.createElement('span');
            cursor.className = 'cursor';
            typewriterElement.appendChild(cursor);
            
            // 打字机效果函数
            function typeWriter() {
                if (i < originalText.length) {
                    // 在光标前插入文本
                    const textNode = document.createTextNode(originalText.charAt(i));
                    typewriterElement.insertBefore(textNode, cursor);
                    i++;
                    setTimeout(typeWriter, speed);
                } else {
                    // 动画完成,隐藏光标
                    setTimeout(() => {
                        cursor.classList.add('hidden');
                    }, 1000); // 在最后一个字符后再等待1秒,然后隐藏光标
                }
            }
            
            // 启动打字机效果
            typeWriter();
        });
        </script>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

现在,摘要区域有了展开/收起功能,默认展开,点击"收起"按钮后会只显示第一行内容。

8. 添加Deepseek签名和链接

最后,我们在摘要底部添加Deepseek R1-Reasoner的署名和链接:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5; transition: all 0.3s ease;">
        <div class="d-flex justify-content-between align-items-center mb-2">
            <h5 class="fw-bold text-primary mb-0">文章摘要</h5>
            <span id="toggle-summary" class="text-primary" style="cursor: pointer; font-size: 0.9rem;">收起</span>
        </div>
        <!-- 隐藏的div用于计算高度 -->
        <div id="hidden-summary" style="visibility: hidden; position: absolute; width: calc(100% - 3rem); color: #2171d3; padding: 0.5rem 0;"><?php echo $this->fields->summary; ?></div>
        <!-- 实际显示的div -->
        <div class="summary-container" id="summary-container">
            <div class="summary-content text-primary" id="typewriter-text" style="color: #2171d3; font-weight: normal; min-height: 1.5em; padding: 0.5rem 0;"></div>
            <div class="text-end mt-2" style="font-size: 0.75rem;">
                <span style="color: #1e88e5;">Powered by</span><span style="display: inline-block; width: 6px;"></span><a href="https://deepseek.com" target="_blank" style="color: #1e88e5; font-weight: bold; text-decoration: underline;">Deepseek R1-Reasoner</a>
            </div>
        </div>
        
        <style>
            .cursor {
                display: inline-block;
                width: 2px;
                height: 1em;
                background-color: #2171d3;
                margin-left: 2px;
                animation: blink 0.7s infinite;
                vertical-align: text-bottom;
            }
            
            @keyframes blink {
                0%, 100% { opacity: 1; }
                50% { opacity: 0; }
            }
            
            .summary-container.collapsed {
                height: 2em; /* 增加高度确保显示完整一行 */
                overflow: hidden;
                margin: 0;
            }
            
            /* 当动画完成时隐藏光标 */
            .cursor.hidden {
                display: none;
            }
        </style>
        
        <script>
        document.addEventListener('DOMContentLoaded', function() {
            // 存储原始内容
            const originalText = `<?php echo addslashes($this->fields->summary); ?>`;
            const typewriterElement = document.getElementById('typewriter-text');
            const hiddenElement = document.getElementById('hidden-summary');
            const summaryContainer = document.getElementById('summary-container');
            const toggleButton = document.getElementById('toggle-summary');
            
            // 展开/收起状态
            let isExpanded = true;
            
            // 根据隐藏元素设置打字机容器高度
            setTimeout(() => {
                const height = hiddenElement.offsetHeight;
                // 增加额外高度容纳底部信息
                typewriterElement.style.height = (height + 10) + 'px';
                // 高度设置完成后可以隐藏计算元素
                hiddenElement.remove();
            }, 0);
            
            // 展开/收起按钮事件
            toggleButton.addEventListener('click', function() {
                isExpanded = !isExpanded;
                if (isExpanded) {
                    summaryContainer.classList.remove('collapsed');
                    toggleButton.textContent = '收起';
                } else {
                    summaryContainer.classList.add('collapsed');
                    toggleButton.textContent = '展开';
                }
            });
            
            let i = 0;
            const speed = 30; // 打字速度,数值越小越快
            
            // 创建光标元素
            const cursor = document.createElement('span');
            cursor.className = 'cursor';
            typewriterElement.appendChild(cursor);
            
            // 打字机效果函数
            function typeWriter() {
                if (i < originalText.length) {
                    // 在光标前插入文本
                    const textNode = document.createTextNode(originalText.charAt(i));
                    typewriterElement.insertBefore(textNode, cursor);
                    i++;
                    setTimeout(typeWriter, speed);
                } else {
                    // 动画完成,隐藏光标
                    setTimeout(() => {
                        cursor.classList.add('hidden');
                    }, 1000); // 在最后一个字符后再等待1秒,然后隐藏光标
                }
            }
            
            // 启动打字机效果
            typeWriter();
        });
        </script>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

完整代码

以下是最终完整的代码实现:

phpCopy<?php else : ?>
    <?php if (isset($this->fields->summary) && $this->fields->summary): ?>
    <div class="article-summary mb-4 p-3 rounded" style="background-color: #e6f3ff; border-left: 4px solid #1e88e5; transition: all 0.3s ease;">
        <div class="d-flex justify-content-between align-items-center mb-2">
            <h5 class="fw-bold text-primary mb-0">文章摘要</h5>
            <span id="toggle-summary" class="text-primary" style="cursor: pointer; font-size: 0.9rem;">收起</span>
        </div>
        <!-- 隐藏的div用于计算高度 -->
        <div id="hidden-summary" style="visibility: hidden; position: absolute; width: calc(100% - 3rem); color: #2171d3; padding: 0.5rem 0;"><?php echo $this->fields->summary; ?></div>
        <!-- 实际显示的div -->
        <div class="summary-container" id="summary-container">
            <div class="summary-content text-primary" id="typewriter-text" style="color: #2171d3; font-weight: normal; min-height: 1.5em; padding: 0.5rem 0;"></div>
            <div class="text-end mt-2" style="font-size: 0.75rem;">
                <span style="color: #1e88e5;">Powered by</span><span style="display: inline-block; width: 6px;"></span><a href="https://deepseek.com" target="_blank" style="color: #1e88e5; font-weight: bold; text-decoration: underline;">Deepseek R1-Reasoner</a>
            </div>
        </div>
        
        <style>
            .cursor {
                display: inline-block;
                width: 2px;
                height: 1em;
                background-color: #2171d3;
                margin-left: 2px;
                animation: blink 0.7s infinite;
                vertical-align: text-bottom;
            }
            
            @keyframes blink {
                0%, 100% { opacity: 1; }
                50% { opacity: 0; }
            }
            
            .summary-container.collapsed {
                height: 2em; /* 增加高度确保显示完整一行 */
                overflow: hidden;
                margin: 0;
            }
            
            /* 当动画完成时隐藏光标 */
            .cursor.hidden {
                display: none;
            }
        </style>
        
        <script>
        document.addEventListener('DOMContentLoaded', function() {
            // 存储原始内容
            const originalText = `<?php echo addslashes($this->fields->summary); ?>`;
            const typewriterElement = document.getElementById('typewriter-text');
            const hiddenElement = document.getElementById('hidden-summary');
            const summaryContainer = document.getElementById('summary-container');
            const toggleButton = document.getElementById('toggle-summary');
            
            // 展开/收起状态
            let isExpanded = true;
            
            // 根据隐藏元素设置打字机容器高度
            setTimeout(() => {
                const height = hiddenElement.offsetHeight;
                // 增加额外高度容纳底部信息
                typewriterElement.style.height = (height + 10) + 'px';
                // 高度设置完成后可以隐藏计算元素
                hiddenElement.remove();
            }, 0);
            
            // 展开/收起按钮事件
            toggleButton.addEventListener('click', function() {
                isExpanded = !isExpanded;
                if (isExpanded) {
                    summaryContainer.classList.remove('collapsed');
                    toggleButton.textContent = '收起';
                } else {
                    summaryContainer.classList.add('collapsed');
                    toggleButton.textContent = '展开';
                }
            });
            
            let i = 0;
            const speed = 30; // 打字速度,数值越小越快
            
            // 创建光标元素
            const cursor = document.createElement('span');
            cursor.className = 'cursor';
            typewriterElement.appendChild(cursor);
            
            // 打字机效果函数
            function typeWriter() {
                if (i < originalText.length) {
                    // 在光标前插入文本
                    const textNode = document.createTextNode(originalText.charAt(i));
                    typewriterElement.insertBefore(textNode, cursor);
                    i++;
                    setTimeout(typeWriter, speed);
                } else {
                    // 动画完成,隐藏光标
                    setTimeout(() => {
                        cursor.classList.add('hidden');
                    }, 1000); // 在最后一个字符后再等待1秒,然后隐藏光标
                }
            }
            
            // 启动打字机效果
            typeWriter();
        });
        </script>
    </div>
    <?php endif; ?>
    <article class="typography" id="post">
        <?= Context::CtxFilter($this) ?>
    </article>

功能特点总结

我们实现的AI打字机文章摘要功能具有以下特点:

  • 美观的蓝色主题:蓝色背景和蓝色文字,视觉效果统一
  • 打字机效果:摘要内容会逐字显示,模拟打字效果
  • 光标闪烁:有一个闪烁的光标跟随文字输入
  • 预设容器高度:避免打字过程中页面跳动
  • 展开/收起功能:可以折叠摘要内容,只显示第一行
  • Deepseek署名:底部有Deepseek R1-Reasoner的署名和链接
  • 自适应布局:在各种屏幕尺寸上都能正常显示

使用方法

  1. 在Typecho后台文章编辑页面底部的自定义字段区域,添加名为"summary"的字段
  2. 在summary字段中填写文章摘要内容
  3. 将完整代码添加到你的文章页面模板中
  4. 发布文章,查看效果

这样,你的博客就拥有了一个带有打字机效果的AI文章摘要功能!

无标签
文章目录