- VisualStudio2022插件的安装及使用-编程手把手系列文章
- pprof-在现网场景怎么用
- C#实现的下拉多选框,下拉多选树,多级节点
- 【学习笔记】基础数据结构:猫树
基于之前的需求,我们的聊天中包含一些简单的交互功能,例如语音转文本:
我们将之前的这几篇结合起来实现该需求:
react-chat-element 实战小计 。
语音转文字 。
雪碧图 。
使用 gka 生成雪碧图,建议将该效果封装为组件 。
安装 gka 工具:
npm install gka -g
生成雪碧图:使用 gka 处理你的动画帧图片,生成雪碧图和 CSS 文件.
gka dir path/to/your/images -o path/to/output -u -m -s
这里,path/to/your/images 是包含你所有帧图片的目录,path/to/output 是输出目录。-u 选项用于相同图片复用优化,-m 用于图片压缩,-s 用于合图优化 。
编写组件代码:在 LoadingSpinner.js 中,引入生成的雪碧图和 CSS 文件,并创建一个组件,该组件使用 div 来显示雪碧图,并通过修改 div 的样式来调整动画的大小 。
import React from 'react';
import spriteSheet from './path/to/output/sprites.png'; // 雪碧图路径
import './path/to/output/gka.css'; // gka 生成的 CSS 文件
const LoadingSpinner = ({ size = '100px' }) => {
return (
<div className="gka-base" style={{ width: size, height: size, backgroundImage: `url(${spriteSheet})` }}></div>
);
};
export default LoadingSpinner;
在这个组件中,size 是一个可选 props,允许动态设置组件的大小.
import { useEffect, useState } from 'react';
import { MessageBox } from 'react-chat-elements';
import 'react-chat-elements/dist/main.css';
import AISVG from '@/assets/imgs/AI.svg';
import UserSVG from '@/assets/imgs/user.svg';
import LoadingSpinner from '../LoadingSpinner';
const MessageList = () => {
const [messageList, setMessageList] = useState([]);
const [maskVisible, setMaskVisible] = useState(false);
const [loadingTimer, setLoadingTimer] = useState(null);
const [textToConfirm, setTextToConfirm] = useState('');
useEffect(() => {
scrollToBottom();
}, [messageList]);
const scrollToBottom = () => {
const msgElement = document.getElementById('message-list');
const scrollTop = msgElement.scrollTop;
const scrollHeight = msgElement.scrollHeight;
const clientHeight = msgElement.clientHeight;
msgElement.scrollTop = scrollHeight - clientHeight;
};
const handleOpenFile = async (e, item) => {
// 文件点击逻辑
};
const addListeningMessgae = () => {
const newMessageList = messageList?.filter((item) => item.key !== 'speech2text');
const newMessage = {
key: 'speech2text',
type: 'text',
content: (
<div className="listening-item">
<div className="listening-item-text">
<LoadingSpinner size="30px" />
{'Please talk, I am listening...'}
</div>
</div>
),
data: {
listening: true
}
};
newMessageList.push(newMessage);
setMessageList(newMessageList);
};
return (
<>
<div className="message-list" id="message-list" onScroll={() => scrollToBottom()}>
<MessageBox
avatar={AISVG}
position="left"
type="text"
text="Hello, I am Univers AI. I am the first professional AI in the sustainable industry, dedicated to helping you optimize your building systems."
className="message-item message-item-left"
/>
<div className="init-list">
{!messageList?.length &&
initContents.map((item, index) => (
<InitComponent title={item?.title} content={item?.content} key={item?.key} index={index} tabKey={item?.key} />
))}
</div>
{messageList.map((item) => (
<MessageBox
key={item?.key}
avatar={item?.isBot ? UserSVG : AISVG}
position={item?.isBot ? 'right' : 'left'}
type={item?.type}
text={item?.type === 'text' ? item?.content : item?.text}
data={item?.data}
onOpen={item?.type === 'photo' || item?.type === 'file' ? (e) => handleOpenFile(e, item) : null}
onDownload={item?.type === 'photo' || item?.type === 'file' ? (e) => handleOpenFile(e, item) : null}
/>
))}
)}
</div>
</>
);
};
export default MessageList;
const addConfirmMessage = () => {
const newMessageList = messageList?.filter((item) => item.key !== 'speech2text');
const newMessage = {
key: 'speech2text',
type: 'text',
content: (
<div className="listening-item">
<div className="listening-item-text">
<LoadingSpinner size="30px" />
{textToConfirm || 'Please talk, I am listening...'}
</div>
{!speaking && renderOptions()}
</div>
),
data: {
listening: true
}
};
newMessageList.push(newMessage);
setMessageList(newMessageList);
};
const renderOptions = () => {
return (
<div className="listening-item-options">
<div className="listening-item-options-item reject" onClick={rejectText}>
×
</div>
<div className="listening-item-options-item confirm" onClick={confirmText}>
√
</div>
</div>
);
};
const rejectText = () => {
// 停止语音识别
recognizer?.stopContinuousRecognitionAsync();
const newMessageList = messageList.filter((item) => item.key !== 'speech2text');
setMessageList(newMessageList);
setTextToConfirm('');
};
const confirmText = () => {
// 停止语音识别
recognizer?.stopContinuousRecognitionAsync();
const newMessageList = [...messageList];
newMessageList[newMessageList.length - 1] = {
key: uuid(),
type: 'text',
content: textToConfirm
};
setMessageList(newMessageList);
setTextToConfirm('');
};
定义组件:在组件中,使用状态来存储当前显示的文本,并使用一个副作用来逐步更新这个状态,模拟打字机的打字效果.
实现打字机逻辑:在 useEffect 中,使用 setInterval 来逐字显示文本,直到文本完成.
清理定时器:确保在组件卸载时清除定时器,避免内存泄漏.
import React, { useState, useEffect } from 'react';
function Typewriter({ text, typingSpeed }) {
const [displayText, setDisplayText] = useState('');
const [currentIndex, setCurrentIndex] = useState(0);
useEffect(() => {
const interval = setInterval(() => {
if (currentIndex < text.length) {
setDisplayText((prevText) => prevText + text[currentIndex]);
setCurrentIndex((prevIndex) => prevIndex + 1);
} else {
clearInterval(interval);
}
}, typingSpeed);
return () => {
clearInterval(interval);
};
}, [text, typingSpeed, currentIndex]);
return <div>{displayText}</div>;
}
export default Typewriter;
。
最后此篇关于构建交互式聊天界面-2的文章就讲到这里了,如果你想了解更多关于构建交互式聊天界面-2的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我正在开发一个使用 JTable 的简单 Java 应用程序。我正在尝试实现我所看到的here 。给定链接上的表格的行为与我在表格中想要的行为完全相同。我已完全按照引用链接上显示的方式进行操作。 但是
我需要将 *.ml 文件加载到 Ocaml 顶层(交互式解释器,当您在 shell 中键入“ocaml”时),然后从 Matlab 进程发送指令,获取指令结果,发送返回另一条指令,... 我编写了这个
我正在尝试实现交互式 UILocalNotifications。 以下是我的代码。我无法获得接收通知的 3 个操作按钮。 UIMutableUserNotificationAction *nActio
我不是一个干净的 CSS 编码器,所以这可能是我问题的症结所在……但是……我从本教程中拼凑了一张交互式 map :http://www.noobcube.com/tutorials/html-css/
这个问题在这里已经有了答案: 关闭 13 年前。 Possible Duplicate: How to save a Python interactive session? 我可以在“头脑 Stor
我试图在不扩展 ListActivity 类的情况下创建交互式 ListView。布局应该是左侧的 RatingBar 和标签向右。这是我目前的代码,没有任何编译错误,但在应用程序启动时崩溃: pub
我正在尝试使用 antlr 编写一种简单的交互式(使用 System.in 作为源)语言,但我遇到了一些问题。我在网上找到的例子都是使用每行循环,例如: while(readline) resul
我想创建一个交互式 JTable。为此,我想在表格的单元格中添加 JPanel。一旦 JPanel 位于单元格中,我就可以将我的各种组件添加到 JPanel 中,从而使表格具有交互性。每个 JPane
我有两个具有完全相同文件的分支(如果您想知道它是一个 .sql 文件),我想以交互方式 merge 它。 我非常想像在发生冲突(或命令行)时那样打开一个 diff 程序,然后准确选择行到哪里。 有什么
我想用交互式 rebase 编辑一个提交。当我用 edit 替换 pick 时,此提交的更改仍应用于 repo(以便我可以编辑提交)但我想从头开始重写它。我该怎么做? 最佳答案 在交互式 rebase
来自 Python relative imports for the billionth time : 要使 from .. import 起作用,模块名称中的点数必须至少与 import 语句中的点
如何使条形与一整天一样宽?现在,条形图更像是划分一天的开始的线。我是否从条形标记切换到矩形标记?我会玩带秤吗?或者我是否在不修改数据的情况下强制分箱? 现在条形图太细了。因为图形是交互式的,所以我无法
是否有任何命令行技巧可以让 SVN 添加来自 svn stat 的所有丢失的文件?互动? 例如,类似于: svn add --interactive $ new file: file1.tmp
我想知道这是否可以(可能不是)使用 R 中的并行处理后端之一来完成。 .我尝试了一些谷歌搜索,但一无所获。 我目前遇到的一般问题: 我有一些大物体需要大约半小时才能到达 load 我想在数据上生成一系
我一直在关注问题 here 的建议暂时将各种提示/信息放在我在 Altair 中的地 block 上。但是,如果 Altair 绘图设置为 interactive(),则此建议不起作用 - 在我看来,
简短的问题是,如果 shell 位于不拥有 tty 的孤立进程组中,它应该做什么?但我建议阅读长问题,因为它很有趣。 这是一种有趣且令人兴奋的方法,可以使用您最喜欢的 shell 将您的笔记本电脑变成
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a software
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
有没有办法实现交互式 cli,允许您从多个选项中进行选择? 我想要实现如下目标: 显示的示例是使用 JavaScript 中的 Quirer.js 库实现的,但我似乎找不到在 Java 中实现类似功能
我正在尝试使用运行一些命令的输入脚本来运行交互式 R (Windows XP),然后让我进入 R 命令行提示符。但是,当我运行它时,它会退出。 例如,这是输入文件: test.r: x = 1 x 以
我是一名优秀的程序员,十分优秀!