Contoh terperinci vue3 merealisasikan kesan mesin taip chatgpt
在做 chatgpt 镜像站的时候,发现有些镜像站是没做打字机的光标效果的,就只是文字输出,是他们不想做吗?反正我想做。于是我仔细研究了一下,实现了打字机效果加光标的效果,现在分享一下我的解决方案以及效果图
共识
首先要明确一点,chatgpt 返回的文本格式是 markdown 的,最基本的渲染方式就是把 markdown 文本转换为 HTML 文本,然后 v-html
渲染即可。这里的转换和代码高亮以及防 XSS 攻击用到了下面三个依赖库:
- marked 将markdwon 转为 html
- highlight 处理代码高亮
- dompurify 防止 XSS 攻击
同时我们是可以在 markdown 中写 html 元素的,这意味着我们可以直接把光标元素放到最后!
将 markdown 转为 html 并处理代码高亮
先贴代码
MarkdownRender.vue
<script setup> import {computed} from 'vue'; import DOMPurify from 'dompurify'; import {marked} from 'marked'; import hljs from '//cdn.staticfile.org/highlight.js/11.7.0/es/highlight.min.js'; import mdInCode from "@/utils/mdInCode"; // 用于判断是否显示光标 const props = defineProps({ // 输入的 markdown 文本 text: { type: String, default: "" }, // 是否需要显示光标?比如在消息流结束后是不需要显示光标的 showCursor: { type: Boolean, default: false } }) // 配置高亮 marked.setOptions({ highlight: function (code, lang) { try { if (lang) { return hljs.highlight(code, {language: lang}).value } else { return hljs.highlightAuto(code).value } } catch (error) { return code } }, gfmtrue: true, breaks: true }) // 计算最终要显示的 html 文本 const html = computed(() => { // 将 markdown 转为 html function trans(text) { return DOMPurify.sanitize(marked.parse(text)); } // 光标元素,可以用 css 美化成你想要的样子 const cursor = '<span></span>'; if (props.showCursor) { // 判断 AI 正在回的消息是否有未闭合的代码块。 const inCode = mdInCode(props.text) if (inCode) { // 有未闭合的代码块,不显示光标 return trans(props.text); } else { // 没有未闭合的代码块,将光标元素追加到最后。 return trans(props.text + cursor); } } else { // 父组件明确不显示光标 return trans(props.text); } }) </script> <template> <!-- tailwindcss:leading-7 控制行高为1.75rem --> <div v-html="html" class="markdown leading-7"> </div> </template> <style> /** 设置代码块样式 **/ .markdown pre { @apply bg-[#282c34] p-4 mt-4 rounded-md text-white w-full overflow-x-auto; } .markdown code { width: 100%; } /** 控制段落间的上下边距 **/ .markdown p { margin: 1.25rem 0; } .markdown p:first-child { margin-top: 0; } /** 小代码块样式,对应 markdown 的 `code` **/ .markdown :not(pre) > code { @apply bg-[#282c34] px-1 py-[2px] text-[#e06c75] rounded-md; } /** 列表样式 **/ .markdown ol { list-style-type: decimal; padding-left: 40px; } .markdown ul { list-style-type: disc; padding-left: 40px; } /** 光标样式 **/ .markdown .cursor { display: inline-block; width: 2px; height: 20px; @apply bg-gray-800 dark:bg-gray-100; animation: blink 1.2s step-end infinite; margin-left: 2px; vertical-align: sub; } @keyframes blink { 0% { opacity: 1; } 50% { opacity: 0; } 100% { opacity: 1; } } </style>
可以发现最基本的 markdown 显示还是挺简单的,话就不多说了,都在注释里。
我想你也许对判断消息中的代码块是否未闭合更感兴趣,那么就继续看下去吧!
代码块是否未闭合
markdown 有两种代码块,一种是 `code` ,另一种是 " code ",我叫他小代码块和大代码块。
一开始我是想用正则去判断的,但是奈何有点复杂,我实在想不出应该如何去编写正则,让 chatgpt 写的正则也会判断失败,而且还要考虑到转义符,就算写出了正则,估计也会很复杂和难以维护。
经过短暂的苦思冥想后,我想到了之前在 《Vue.js设计与实现》 中看到的用有限元状态机解析 html 文本的方案。
这里有请 chatgpt 简单介绍一下有限元状态机:
有限元状态机(finite element state machine)是一种基于状态机的建模技术,它可以将系统的行为描述为一组状态和状态之间的转换。每个状态可以对应系统内的一种状态,而状态之间的转换则表示系统内部的转换过程。有限元状态机还可以用于系统的可靠性分析和测试,同时也可以作为软件工程中的一种建模工具,帮助代码实现、测试和调试等方面。
在实际应用中,有限元状态机通常是通过自动化工具生成,并且可以用于控制系统、网络通信协议、软件接口等方面的设计和测试。由于有限元状态机具有自动化生成和完备性检查等优点,因此它已经成为了一种广泛应用的建模与测试工具。
请站在web前端的视角下进行介绍
从前端开发的角度来看,有限元状态机可以用于构建复杂的交互式Web应用程序。在Web应用程序中,有限元状态机可以描述用户界面的不同状态(例如,登录、注册、未登录等),以及状态之间的转换(例如,从登录状态转换到注册状态)。
通常情况下,前端工程师使用JavaScript实现有限元状态机,包括定义状态、定义状态之间的转换规则、处理状态转换时的行为等。使用有限元状态机可以帮助前端工程师更好地掌控应用程序的状态,从而减少代码中的复杂条件分支和无法预测的行为。同时,有限元状态机也可以帮助前端团队共同理解应用程序的状态和转换规则,从而更好地协作开发和维护Web应用程序。
总之,有限元状态机是一种非常有用的前端开发技术,可以帮助前端工程师更好地构建和管理Web应用程序的状态和行为,提高应用程序的可靠性和用户体验。
回到正题,我可以一点一点的从头开始去解析 markdown 文本。想象这么一个简单的状态转换流程:
- 初始状态为文本状态。
- 遇到代码块标记,文本状态转换到代码块开始状态。
- 再次遇到代码块标记,从代码块开始状态转换到文本状态。
不过现实要更复杂一点,我们有小代码块和大代码块。有限元状态机的妙处就在这里,当处在小代码块状态的时候,我们不需要操心大代码块和正常文本的事,他的下一个状态只能是遇到小代码块的闭合标签,进入文本状态。
理解了这些,再来看我的源码,才会发现他的精妙。
const States = { text: 0, // 文本状态 codeStartSm: 1, // 小代码块状态 codeStartBig: 2, // 大代码块状态 } /** * 判断 markdown 文本中是否有未闭合的代码块 * @param text * @returns {boolean} */ function isInCode(text) { let state = States.text let source = text let inStart = true // 是否处于文本开始状态,即还没有消费过文本 while (source) { // 当文本被解析消费完后,就是个空字符串了,就能跳出循环 let char = source.charAt(0) // 取第 0 个字 switch (state) { case States.text: if (/^\n?```/.test(source)) { // 以 ``` 或者 \n``` 开头。表示大代码块开始。 // 一般情况下,代码块前面都需要换行。但是如果是在文本的开头,就不需要换行。 if (inStart || source.startsWith('\n')) { state = States.codeStartBig } source = source.replace(/^\n?```/, '') } else if (char === '\\') { // 遇到转义符,跳过下一个字符 source = source.slice(2) } else if (char === '`') { // 以 ` 开头。表示小代码块开始。 state = States.codeStartSm source = source.slice(1) } else { // 其他情况,直接消费当前字符 source = source.slice(1) } inStart = false break case States.codeStartSm: if (char === '`') { // 遇到第二个 `,表示代码块结束 state = States.text source = source.slice(1) } else if (char === '\\') { // 遇到转义符,跳过下一个字符 source = source.slice(2) } else { // 其他情况,直接消费当前字符 source = source.slice(1) } break case States.codeStartBig: if (/^\n```/.test(source)) { // 遇到第二个 ```,表示代码块结束 state = States.text source = source.replace(/^\n```/, '') } else { // 其他情况,直接消费当前字符 source = source.slice(1) } break } } return state !== States.text } export default isInCode
到这里,就已经实现了一个 chatgpt 消息渲染了。喜欢的话点个赞吧!谢谢!
Atas ialah kandungan terperinci Contoh terperinci vue3 merealisasikan kesan mesin taip chatgpt. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool
Gambar buka pakaian secara percuma

Clothoff.io
Penyingkiran pakaian AI

AI Hentai Generator
Menjana ai hentai secara percuma.

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas



DALL-E 3 telah diperkenalkan secara rasmi pada September 2023 sebagai model yang jauh lebih baik daripada pendahulunya. Ia dianggap sebagai salah satu penjana imej AI terbaik setakat ini, mampu mencipta imej dengan perincian yang rumit. Walau bagaimanapun, semasa pelancaran, ia adalah tidak termasuk

Langkah pemasangan: 1. Muat turun perisian ChatGTP dari laman web rasmi ChatGTP atau kedai mudah alih 2. Selepas membukanya, dalam antara muka tetapan, pilih bahasa sebagai bahasa Cina 3. Dalam antara muka permainan, pilih permainan mesin manusia dan tetapkan Spektrum bahasa Cina; 4 Selepas memulakan, masukkan arahan dalam tetingkap sembang untuk berinteraksi dengan perisian.

PHP dan Vue: gandingan sempurna alat pembangunan bahagian hadapan Dalam era perkembangan pesat Internet hari ini, pembangunan bahagian hadapan telah menjadi semakin penting. Memandangkan pengguna mempunyai keperluan yang lebih tinggi dan lebih tinggi untuk pengalaman tapak web dan aplikasi, pembangun bahagian hadapan perlu menggunakan alat yang lebih cekap dan fleksibel untuk mencipta antara muka yang responsif dan interaktif. Sebagai dua teknologi penting dalam bidang pembangunan bahagian hadapan, PHP dan Vue.js boleh dianggap sebagai alat yang sempurna apabila digandingkan bersama. Artikel ini akan meneroka gabungan PHP dan Vue, serta contoh kod terperinci untuk membantu pembaca memahami dan menggunakan kedua-dua ini dengan lebih baik.

chatgpt boleh digunakan di China, tetapi tidak boleh didaftarkan, begitu juga di Hong Kong dan Macao Jika pengguna ingin mendaftar, mereka boleh menggunakan nombor telefon mudah alih asing untuk mendaftar. Perhatikan bahawa semasa proses pendaftaran, persekitaran rangkaian mesti ditukar IP asing.

Dalam temu bual pembangunan bahagian hadapan, soalan lazim merangkumi pelbagai topik, termasuk asas HTML/CSS, asas JavaScript, rangka kerja dan perpustakaan, pengalaman projek, algoritma dan struktur data, pengoptimuman prestasi, permintaan merentas domain, kejuruteraan bahagian hadapan, corak reka bentuk, dan teknologi dan trend baharu. Soalan penemuduga direka bentuk untuk menilai kemahiran teknikal calon, pengalaman projek dan pemahaman tentang trend industri. Oleh itu, calon harus bersedia sepenuhnya dalam bidang ini untuk menunjukkan kebolehan dan kepakaran mereka.

Django ialah rangka kerja aplikasi web yang ditulis dalam Python yang menekankan pembangunan pesat dan kaedah bersih. Walaupun Django ialah rangka kerja web, untuk menjawab soalan sama ada Django ialah front-end atau back-end, anda perlu mempunyai pemahaman yang mendalam tentang konsep front-end dan back-end. Bahagian hadapan merujuk kepada antara muka yang pengguna berinteraksi secara langsung, dan bahagian belakang merujuk kepada program bahagian pelayan Mereka berinteraksi dengan data melalui protokol HTTP. Apabila bahagian hadapan dan bahagian belakang dipisahkan, program bahagian hadapan dan bahagian belakang boleh dibangunkan secara bebas untuk melaksanakan logik perniagaan dan kesan interaktif masing-masing, dan pertukaran data.

Sebagai pembangun C#, kerja pembangunan kami biasanya merangkumi pembangunan bahagian hadapan dan bahagian belakang Apabila teknologi berkembang dan kerumitan projek meningkat, pembangunan kolaboratif bahagian hadapan dan bahagian belakang menjadi semakin penting dan kompleks. Artikel ini akan berkongsi beberapa teknik pembangunan kolaboratif bahagian hadapan dan belakang untuk membantu pembangun C# menyelesaikan kerja pembangunan dengan lebih cekap. Selepas menentukan spesifikasi antara muka, pembangunan kolaboratif bahagian hadapan dan belakang tidak dapat dipisahkan daripada interaksi antara muka API. Untuk memastikan pembangunan kolaboratif bahagian hadapan dan belakang yang lancar, perkara yang paling penting ialah menentukan spesifikasi antara muka yang baik. Spesifikasi antara muka melibatkan nama antara muka

Open AI akhirnya membuat cariannya. Syarikat San Francisco baru-baru ini telah mengumumkan alat AI baharu dengan keupayaan carian. Pertama kali dilaporkan oleh The Information pada Februari tahun ini, alat baharu ini dipanggil SearchGPT dan menampilkan c
