AI对话系统单页html源码_单文件GPT畅聊 v2源码

图片[1]-AI对话系统单页html源码_单文件GPT畅聊 v2源码-晨风乐享网

v2版本
增加自定义提示词支持,优化网页整体布局,修复system_prompt导致的消息发送bug。
新增几个通用且好用的的提示词,仅供参考。

注意,如果您在模型列表中配置了自定义prompt,在对应模型被选中时,该prompt将会覆盖system_prompt。

单html文件ChatGPT网站,支持全平台使用,电脑手机点击即用。
使用第三方转发api进行问答,您可以使用github账号申请免费的key进行使用。具体请看源代码内的说明。
在文件第八行后填入您申请的apikey,然后调整模型,即可使用、分享。
请注意,apikey直接硬编码进入了网页内部,填入您自己的key后,不可将该网页公开传播,防止apikey滥用。

代码无图床依赖,js、css均在html文件内,图片也使用base64直接编码进入了网页。
markdown部分的js和css使用了由CDN托管的markdown-it进行渲染,确保在有网情况下即可使用。(如果报错Uncaught TypeError: window.markdownit is not a constructor,说明从该CDN下载文件失败,可稍后尝试)
除去公共CDN的公开文件外,代码完全单文件,无需任何依赖即可使用。

<!DOCTYPE html>
<html lang="zh-CN">

<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>简易ChatGPT</title>
	<script>
		// AI配置
		var api = "https://api.gpt.ge/v1/chat/completions"; // 您的api地址
		var apikey = ""; // 您的api-key

		var system_prompt = null; // 系统提示词,可留空
		var llm_selection = [
			["gpt-4o-2024-08-06", "GPT-4o"],
			["gpt-4o-mini", "GPT-4o-mini"],
			["o1-sim", "[+]拟造GPT o1"],
			["o1-mini-sim", "[+]拟造GPT o1 mini"],
			["thinker", "[+]深入思考"],
			["prompt_expert", "[+]提示词工程师"],
			["o1-preview", "GPT o1 preview"],
			["o1-mini", "GPT o1 mini"],
			["gpt-4o-all", "[全能]GPT-4o-all"],
			["gpt-4-turbo", "GPT-4-Turbo"],
			["net-gpt-4-turbo", "[联网]GPT-4-Turbo"],
			["claude-3-5-sonnet-20240620", "Claude-3.5"],
			["deepseek-chat", "Deepseek Chat"],
			["deepseek-coder", "Deepseek Coder"],
			["gemini-1.5-flash-latest", "[长文]Gemini Flash"],
			["gemini-1.5-pro-latest", "[长文]Gemini Pro"],
			["stable-diffusion", "[绘图]Stable Diffusion"],
		]; // API可用模型配置,每行第一个是模型代码,第二个是模型在下拉菜单中显示的内容。注意尾部逗号。
		// 如果使用prompt,请在第一列自行配置一个名字,然后在下方 “prompt_model_list” 数组中,配置prompt和真实模型。
		
		// 网页配置
		var isinfo = 1;// 是否有开屏提示信息
		var webFontFamily = "华文中宋, 微软雅黑, 楷体";// 网页全局字体,多个字体用英文逗号隔开,越靠前优先级越高
		
		// 使用额外变量保存的prompt。请注意,prompt两边使用单引号进行引用,因此内部的文本出现的单引号需要转义。
		var o1_prompt = '# 语言: zh-CN\n# 用途: 逐步解释推理过程,输出为 Markdown 格式\n\ndef 分析助理():\n    """你是一个擅长逐步解释推理过程的AI助手"""\n    return {\n        "风格": ["理性", "细致", "批判性思维"],\n        "擅长": "多步骤推理",\n        "输出格式": "Markdown"\n    }\n\nclass 推理助手(输入):\n    def __init__(self, 输入):\n        self.状态 = "确认用户问题" # 初始化第1步\n        self.输入 = 输入\n\n    def 逐步推理(self):\n        """你会逐步解释每一步的推理过程并提供结论"""\n        \n        def 标题(状态):\n            """为每一步推理生成标题,包含对替代答案的探索。考虑你可能出错的情况,以及如果推理错误,错误可能出现在哪里。"""\n            return 标题\n\n        def 内容描述(状态, 输入):\n            """进行认真细致的推理,注意你作为llm的局限性以及你能做什么和不能做什么。使用最佳实践。"""\n            return 推理过程\n\n        def 决定下一步(状态, 输入, 当前步骤):\n            """根据状态、输入和当前步骤动态决定下一步"""\n            # 至少要有 3 步推理\n            if 当前步骤 >= 3 and 是否可以得出结论(状态, 输入) or 当前步骤 >=  8: \n                return "最终结论"\n            else:\n                return 生成下一步(状态, 输入)\n\n        def 是否可以得出结论(状态, 输入):\n            """判断是否已经可以得出结论"""\n            return True or False\n\n        def 生成下一步(状态, 输入):\n            """根据输入和当前推理步骤生成下一步推理"""\n            return 下一步推理\n\n        当前步骤 = 0\n        \n        md_output = "# 推理过程\n"\n        while self.状态 != "最终结论":\n            当前步骤 += 1\n            next_action = 决定下一步(self.状态, self.输入, 当前步骤)\n            \n            md_output += f"## 步骤: {标题(self.状态)}\n"\n            md_output += f"- **内容**: {内容描述(self.状态, self.输入)}\n"\n            if next_action != "最终结论":\n                md_output += f"- **下一步**: {next_action}\n\n"\n            \n            self.状态 = next_action\n        \n        return md_output\n\ndef start():\n    """启动时运行"""\n    system_role = 分析助理()\n    输入 = input()\n    助手 = 推理助手(输入)\n    结果 = 助手.逐步推理()\n\n    print(结果)\n\n\nif __name__ == "__main__":\n    start()\n\n# 请按照伪代码的规则运行,直接执行 main。**不要尝试解释代码**';
		var prompt_expert_prompt = '你是一个专家级ChatGPT提示工程师,在各种主题方面具有专业知识。在我们的互动过程中,你会称我为“您”。让我们合作创建最好的ChatGPT响应我提供的提示。我们将进行如下交互:\n1.我会告诉你如何帮助我。\n2.根据我的要求,您将建议您应该承担的其他专家角色,除了成为专家级ChatGPT提示工程师之外,以提供最佳响应。然后,您将询问是否应继续执行建议的角色,或修改它们以获得最佳结果。\n3.如果我同意,您将采用所有其他专家角色,包括最初的Expert ChatGPT Prompt Engineer角色\n4.如果我不同意,您将询问应删除哪些角色,消除这些角色,并保留剩余的角色,包括专家级ChatGPT Prompt工程师角色,然后再继续。\n5.您将确认您的活动专家角色,概述每个角色下的技能,并询问我是否要修改任何角色。\n6.如果我同意,您将询问要添加或删除哪些角色,我将通知您。重复步骤5,直到我对角色满意为止。\n7.如果我不同意,请继续下一步。\n8.你会问:“我怎样才能帮助[我对步骤1的回答]?\n9.我会给出我的答案\n10.你会问我是否想使用任何参考来源来制作完美的提示\n11.如果我同意,你会问我想使用的来源数量。\n12.您将单独请求每个来源,在您查看完后确认,并要求下一个。继续,直到您查看了所有源,然后移动到下一步。\n13.您将以列表格式请求有关我的原始提示的更多细节,以充分了解我的期望。\n14.我会回答你的问题。\n15.从这一点开始,您将在所有确认的专家角色下操作,并使用我的原始提示和步骤14中的其他细节创建详细的ChatGPT提示。提出新的提示并征求我的反馈。\n16.如果我满意,您将描述每个专家角色的贡献以及他们将如何协作以产生全面的结果。然后,询问是否缺少任何输出或专家。\n16.1.如果我同意,我将指出缺少的角色或输出,您将在重复步骤15之前调整角色。\n16.2.如果我不同意,您将作为所有已确认的专家角色执行提供的提示,并生成步骤15中概述的输出。继续执行步聚20。\n17.如果我不满意,你会问具体问题的提示\n18.我将提供补充资料。\n19.按照步聚15中的流程生成新提示,并考虑我在步聚18中的反馈\n20.完成回复后,询问我是否需要任何更改。\n21.如果我同意,请请求所需的更改,参考您之前的回复,进行所需的调整,并生成新的提示。重复步骤15-20,直到我对提示符满意为止。';
		var thinker_prompt = '你是一个喜欢深入探索问题的人,需要详细且逻辑清晰的回答。\n你擅长使用费曼学习法和第一性原理来回答问题,并且你拥有强大的批判性思维。\n如果你对答案不确定,先给我一个警告,然后再尽力回答。请按照以下步骤依次完成执行,每回答一个步骤,请告诉我还有几个步骤完成最终结果,给我一个进度条提示。\n每进入到一个步骤,请使用markdown格式加粗标题信息,比如“定义问题”“构建思想之树”字体需要加粗,并自带分隔符,使得信息有明确的分隔:\n1.定义问题:首先当我提出一个问题时,你首先要采用提问的方式告诉我,你认为还需要的最关键的多个信息,要求我考虑各种因素获得更清晰的信息,比如目标受众等相关信息,基于此来定义问题。\n2.构建思想之树:结合我的回答和多角度分析请为我生成不低于五个的解决方案,请以表格的方式输出。\n3.评估阶段:对于每个提出的解决方案,评估其潜在成功可能性。请考虑优点和缺点、需要的初始努力、实施的难度、可能的挑战以及预期的结果。根据这些因素,为每个选项分配一个成功的百分比概率。同样请以表格的方式输出。\n4.扩展阶段:对于每个解决方案,深入思考过程,生成潜在的场景、实施策略、需要的合作伙伴或资源,以及如何克服可能的障碍。同时,考虑任何可能出现的意外结果,以及如何处理他们,同时进一步优化所有方法,目的提高成功概率。\n5.决策决断:根据评估和场景,按照成功概率高低的顺序排列解决方案。为每个排名提供理由,并提供每个解决方案的最后思考或考虑因素。\n倒数第二步,提醒我,下一步降输出规划建议。\n当我回复后,为我输出一个最初提问问题后的最优选择,选择可是多个解决方案的融合方案。\n输出最优选择的结论必须使用表格的方式清晰的呈现和展示其名称、关键任务,对应的目的。\n同时,再使用OKR的方式进行另一个表格输出,描述目标,以及对应目标可量化的部分,方便我进一步落地和执行。\nOKR的进一步解释:\nO是你通过整体方案的回复后整体理解的一个或者多个目标,目标不得超过3个。\n每个O对应得KR不能低于2个,且不超过4个。\nKR必须是能直接实现目标的;\nKR必须是具有进取心、敢创新的,可以是不常规的;KR必须是以产出或者结果为基础的、可衡量的,设定评分标准:KR必须是和时间相练习的。\n最后一步,回顾最初我提出的问题,结合你所有分析结果和规划建议后,给出精准的解决方案。';
		
		// 配置prompt,第一列是你为模型命的名,第二列是真实使用的模型,第三列是提示词,建议使用变量引入。
		var prompt_model_list = [
			["o1-sim", "gpt-4o-2024-08-06", o1_prompt],
			["o1-mini-sim", "gpt-4o-mini", o1_prompt],
			["prompt_expert", "gpt-4o-2024-08-06", prompt_expert_prompt],
			["thinker", "gpt-4o-2024-08-06", thinker_prompt],
		]
		
	</script>
	<!--
	API说明
	Github上有许多官方转发API,本站使用这些转发API进行工作。如果运行此网页的机器使用了代理,也可以直接使用官方API进行GPT问答。
	推荐一些转发API的申请点(注册超过7天的Github账号可申请免费apikey,具体请看对应页面)
	
	[V3 API][推荐]:https://github.com/popjane/free_chatgpt_api
		- 免费Key申请地址:https://free.gpt.ge/github
		- API地址:(免费)https://free.gpt.ge/v1/chat/completions
				  (付费)https://api.gpt.ge/v1/chat/completions
		- 付费API购买地址:https://shop.gpt.ge/
		- 可用模型列表:https://api.gpt.ge/pricing
		- 额外说明:
		  - V3 API相对于ChatAnyWhere价格更加便宜,推荐购买;
		  - V3 API支持非常多的模型,除去官方的gpt系列,claude、gemini、deepseek等模型也可以使用OpenAI的方式进行请求,仅需要更改请求体中的model;
		  - V3 API支持当前最新的OpenAI o1 preview和o1 mini模型,尽管价格稍贵。
		
	[ChatAnyWhere]:https://github.com/chatanywhere/GPT_API_free
		- 免费Key申请地址:https://api.chatanywhere.org/v1/oauth/free/github/render
		- API地址:https://api.chatanywhere.tech/v1/chat/completions
		- 付费API购买地址:https://buyca.shop/
	-->
	<style>
		/*初始化*/
		* {
			margin: 0;
			padding: 0;
		}

		.no-select {
			-webkit-user-select: none;
			-moz-user-select: none;
			-ms-user-select: none;
			user-select: none;
		}

		.clearfix:after {
			clear: both;
		}

		.clearfix:before,
		.clearfix:after {
			content: "";
			display: table;
		}

		body {
			background-color: rgb(238, 237, 241);
		}

		.container {
			height: 100vh;
			width: 90vw;
			margin: 0 auto;
		}

		/*遮罩卡*/
		.info-cover {
			display: none;
			position: absolute;
			top: 0;
			left: 0;
			height: 100%;
			width: 100%;
			background-color: rgb(0, 0, 0, 0.4);
		}

		.info-container {
			height: 400px;
			width: 300px;
			padding: 30px;
			margin: 17vh auto;
			background-color: #fff;
		}

		.info-nav {
			border-bottom: 1px solid #444;
		}

		.info-name {
			margin-top: 0;
		}

		.info-tail {
			text-align: center;
			margin: 10px 0;
			font-size: 16px;
			color: #777;
		}

		.info-content {
			line-height: 25px;
			margin: 15px 0;
		}

		span {
			color: red;
		}

		/*导航栏*/
		.nav {
			text-align: center;
			height: 60px;
			border-bottom: 1px solid #aaa;
			overflow: hidden;
		}

		select {
			display: inline-block;
			text-align: center;
			background-color: rgb(238, 237, 241);
			margin: 10px 5px;
			height: 40px;
			padding: 0 5px;
			font-size: 18px;
			line-height: 40px;
			border-radius: 10px;
			outline: none;
			-webkit-appearance: none;
			-moz-appearance: none;
			appearance: none;
		}

		.icon {
			float: left;
			height: 80%;
			width: auto;
			margin-top: 5px;
		}


		.navlink,
		.navbutton {
			float: right;
			text-decoration: none;
			height: 40px;
			width: 40px;
			line-height: 40px;
			text-align: center;
			margin: 10px 10px;
			background-image: url("");
			background-position: center;
			background-repeat: no-repeat;
			background-size: 100%;
			border: 0;
			border-radius: 12px;
			color: #fff;
			font-size: 20px;
			cursor: default;
		}

		.navlink:hover,
		.navbutton:hover {
			filter: brightness(80%);
		}

		.navlink:active,
		.navbutton:active {
			filter: brightness(60%);
		}

		/*聊天区*/
		.chat-container {
			padding: 10px 0;
			height: calc(100% - 120px - 4px - 20px);
			overflow-y: auto;
		}

		ol,
		ul {
			margin-top: 5px;
			margin-left: 25px;
		}

		table {
			border-collapse: collapse;
		}

		table,
		th,
		td {
			border: 1px solid #000;
		}

		th,
		td {
			padding: 8px;
			text-align: left;
		}

		th {
			background-color: #f2f2f2;
		}

		tr:nth-child(even) {
			background-color: #f9f9f9;
		}

		tr:nth-child(odd) {
			background-color: #fff;
		}

		p {
			margin-top: 5px;
		}

		img {
			width: 100%;
			height: auto;
		}

		h1,
		h2,
		h3,
		h4,
		h5,
		h6 {
			margin-top: 10px;
		}

		pre {
			background-color: rgb(31, 31, 31);
			border-radius: 10px;
			padding: 10px;
			margin-top: 6px;
		}

		code {
			padding-left: 4px;
			padding-right: 4px;
			border-radius: 4px !important;
			color: #000;
			background-color: #ddd;
			font-family: "consolas";
			font-size: 16px;
			border-radius: 10px;
			white-space: pre-wrap;
			word-wrap: break-word;
		}

		pre>code {
			padding-left: 0;
			padding-right: 0;
			background-color: rgb(31, 31, 31);
			border-radius: 0;
			color: #fff;
		}

		.row {
			width: 100%;
		}

		.gpt-icon {
			float: left;
			background-image: url("");
			background-position: 0;
			background-size: contain;
			background-repeat: no-repeat;
			width: 40px;
			height: 40px;
		}

		.gpt-msg {
			position: relative;
			float: left;
			background-color: rgb(255, 255, 255);
			border-radius: 2px 12px 12px 12px;
			padding: 10px;
			margin: 5px 10px;
			max-width: calc(100% - 40px - 40px - 48px);
			word-wrap: break-word;
			overflow-wrap: break-word;
			white-space: normal;
			font-size: 16px;
		}

		.gpt-msg> :first-child {
			margin-top: 0;
		}

		.user-msg> :first-child {
			margin-top: 0;
		}

		.gpt-msg-toolbox {
			float: left;
		}

		.msg-copy {
			float: left;
			content: "";
			background-image: url("");
			width: 15px;
			height: 15px;
			margin: 17px 0;
			padding: 3px;
			border-radius: 3px;
			background-position: 0;
			background-repeat: no-repeat;
			background-size: contain;
		}

		.msg-copy:hover {
			background-color: #aaa;
		}

		.msg-copy:active {
			background-color: #777;
		}

		.msg-withdraw {
			border-radius: 2px;
			float: left;
			content: "";
			background-image: url("");
			width: 15px;
			height: 15px;
			margin: 17px 0;
			padding: 3px;
			border-radius: 3px;
			background-position: 0;
			background-repeat: no-repeat;
			background-size: contain;
		}

		.msg-withdraw:hover {
			background-color: #aaa;
		}

		.msg-withdraw:active {
			background-color: #777;
		}


		.user-icon {
			float: right;
			background-image: url("");
			background-position: 0;
			background-size: contain;
			background-repeat: no-repeat;
			width: 40px;
			height: 40px;
		}

		.user-msg {
			position: relative;
			float: right;
			background-color: rgb(12, 170, 130);
			border-radius: 12px 2px 12px 12px;
			padding: 10px;
			margin: 10px;
			max-width: calc(100% - 40px - 40px - 48px);
			word-wrap: break-word;
			overflow-wrap: break-word;
			white-space: normal;
			font-size: 16px;
		}

		.user-msg-toolbox {
			float: right;
		}

		/*输入区*/
		.input-container {
			border-top: 1px solid #888;
			height: 60px;
		}

		.messagebox {
			display: inline-block;
			height: 3.5vh;
			margin: 1.25vh 10px;
			font-size: 16px;
		}

		.submitbutton {
			float: right;
			text-decoration: none;
			height: 40px;
			width: 60px;
			line-height: 40px;
			text-align: center;
			margin: 10px 0;
			background-color: rgb(12, 170, 130);
			border: 0;
			border-radius: 12px;
			color: #fff;
			font-size: 20px;
			cursor: default;

		}

		.submitbutton:hover {
			background-color: rgb(9, 129, 99);
		}

		.submitbutton:active {
			background-color: rgb(5, 81, 62);
		}

		.msginput {
			float: left;
			height: 30px;
			margin: 10px 10px;
			padding: 5px;
			font-size: 18px;
			outline: none;
			width: calc(100% - 100px);
			outline: none;
			box-shadow: none;
			border: 1px solid #aaa;
			border-radius: 10px;
		}
	</style>
	<script src="https://cdn.jsdelivr.net/npm/markdown-it/dist/markdown-it.min.js"></script>
	<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/sindresorhus/github-markdown-css/github-markdown.css">
	<script>
		var historys;			   // 聊天历史记录嵌套数组
		var myHeaders = new Headers();
		myHeaders.append("Authorization", "Bearer " + apikey);
		myHeaders.append("Content-Type", "application/json");
		var cansubmit = 0;		  // 是否可以提交的信号,防止用户点击多次提交按钮,导致异步出错
		var issystem = 0;		   // 是否有system prompt,会在网页加载完成后自动初始化
		var cnt = 0;				// 网页按钮计数器
		function select_model(model_name) {
			for (i = 0; i < prompt_model_list.length; i++) {
				if (model_name == prompt_model_list[i][0]) {
					return prompt_model_list[i][1]
				}
			}
			return model_name
		}
		function select_prompt(model_name, prompts) {
			for (i = 0; i < prompt_model_list.length; i++) {
				if (model_name == prompt_model_list[i][0]) {
					return prompt_model_list[i][2]
				}
			}
			return prompts
		}
		function cleanhistory() {
			var model = document.getElementById("model").value;
			historys = [];
			cnt = 0;
			selected_prompt = select_prompt(model, system_prompt)
			if (selected_prompt != null) {
				cnt++;
				issystem = 1;
				historys.push({ 'role': 'system', 'content': selected_prompt })
			}
			else {
				issystem = 0;
			}
			cansubmit = 1;
			document.getElementsByClassName('chat-container')[0].innerHTML = '';
		}
		function indisplay() { document.getElementById('infocard').style.display = 'none'; }
		function copyToClipboard(text) {
			if (navigator.clipboard && window.isSecureContext) {
				return navigator.clipboard.writeText(text).then(function () {
				}, function (err) {
					alert("内容复制失败");
				});
			} else {
				const textArea = document.createElement("textarea");
				textArea.value = text;
				document.body.appendChild(textArea);
				textArea.focus();
				textArea.select();
				try {
					document.execCommand('copy');
				} catch (err) {
					alert("内容复制失败");
				}
				document.body.removeChild(textArea);
			}
		}

		window.addEventListener('load', function () {
			//初始化,清空历史
			cleanhistory();
			//监听输入框的回车按键,自动提交
			document.getElementById('user-input').addEventListener('keydown', function (event) {
				if (event.key === 'Enter') {
					if (!event.shiftKey && !event.ctrlKey) {
						submit();
						event.preventDefault();
					}
				}
			});
			//将用户llm填入下拉框
			selects = document.getElementById("model");
			for (i = 0; i < llm_selection.length; i++) { selects.innerHTML += '<option value="' + llm_selection[i][0] + '">' + llm_selection[i][1] + '</option>'; }
			//根据用户配置显示开屏提示
			if (isinfo == 1) { document.getElementById('infocard').style.display = 'block'; }
			//根据用户配置更改全局字体
			document.body.style.fontFamily = webFontFamily;
			document.getElementById("user-input").style.fontFamily = webFontFamily;
		});
		function chat() {
			cansubmit = 0;
			document.getElementsByClassName('chat-container')[0].innerHTML += '<div class="row clearfix"><div class="gpt-icon"></div><div class="gpt-msg"></div><div class="gpt-msg-toolbox"><div class="msg-copy" onclick="copy(' + String(cnt) + ')"></div><div class="msg-withdraw" onclick="gpt_withdraw(' + String(cnt++) + ')"></div></div></div>';
			ele = document.getElementsByClassName('gpt-msg')
			gpt_content = ele[ele.length - 1];
			gpt_content.innerText = "正在思考中,请稍后……";

			raw = JSON.stringify({
				'model': select_model(document.getElementById('model').value),
				'messages': historys
			});
			//组装、发送请求
			var requestOptions = {
				method: 'POST',
				headers: myHeaders,
				body: raw,
				redirect: 'follow'
			};
			fetch(api, requestOptions)
				.then(response => response.json())
				.then(result => {
					if (result.hasOwnProperty("error")) {
						gpt_content.innerText = "请求出现错误,错误信息:\n" + result.error.code + "\n" + result.error.message + "\n请稍后点击[清空聊天]后重试。";
						return 0;
					}
					msglist = result.choices;
					gpt_msg = msglist[msglist.length - 1].message.content;
					historys.push({ "role": "assistant", "content": gpt_msg });
					const md = new window.markdownit();
					gpt_msg_md = md.render(gpt_msg);
					gpt_content.innerHTML = gpt_msg_md;
					cansubmit = 1;
					const rows = document.querySelectorAll('.row');
					rows.forEach(row => {
						const links = row.querySelectorAll('a');
						links.forEach(link => {
							link.target = '_blank';
						});
					});
				})
				.catch(error => {
					console.log(error);
				});
		}
		function submit() {
			if (cansubmit == 0) {
				return 0;
			}
			//获取用户的数据
			var msg = document.getElementById('user-input').value;
			if (msg == "") {
				return 0;
			}
			//更新网页显示
			const md = new window.markdownit();
			msg_md = md.render(msg);
			document.getElementsByClassName('chat-container')[0].innerHTML += '<div class="row clearfix"><div class="user-icon"></div><div class="user-msg"></div><div class="user-msg-toolbox"><div class="msg-copy" onclick="copy(' + String(cnt) + ')"></div><div class="msg-withdraw" onclick="user_withdraw(' + String(cnt++) + ')"></div></div></div>';
			ele = document.getElementsByClassName('user-msg')
			ele[ele.length - 1].innerHTML = msg_md;
			document.getElementById('user-input').value = ""
			//更新历史记录
			historys.push({ "role": "user", "content": msg });
			chat();
		}
		function copy(i) {
			copyToClipboard(historys[i].content)
		}
		function gpt_withdraw(i) {
			if (cansubmit == 0) {
				return 0;
			}
			historys = historys.slice(0, i);
			rows = document.getElementsByClassName('row');
			len = rows.length;
			for (c = len - 1; c >= i - issystem; c--) {
				rows[c].remove();
			}
			cnt = i;
			chat();
		}
		function user_withdraw(i) {
			historys = historys.slice(0, i);
			rows = document.getElementsByClassName('row');
			len = rows.length;
			console.log(len + "长度");
			for (c = len - 1; c >= i - issystem; c--) {
				rows[c].remove();
			}
			cnt = i;
		}
	</script>
</head>

<body>
	<div class="info-cover" id="infocard" onclick="indisplay()">
		<div class="info-container">
			<div class="info-nav">
				<h1 class="info-name">提示</h1>
			</div>
			<div class="info-content">
				<p class="info-content">  本网页仅个人使用,apikey直接硬编码进入了单网页中,<span>切勿在不信任的渠道进行传播、分享</span>,谨防apikey滥用!</p>
				<p class="info-content">  网页采用单文件设计,与markdown相关的依赖文件直接使用了公开CDN进行引用,图片也使用base64直接编码进入了本文件。</p>
				<p class="info-content">  您如果需要修改相关的配置信息,如BaseURL、api-key、可用模型,可以直接修改此网页的源代码。相关配置信息在本文件的第8行之后。</p>
			</div>
			<div class="info-tail">点击页面任意位置关闭</div>
		</div>
	</div>
	<div class="container">
		<div class="nav">
			<svg t="1722656538535" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
				p-id="5538" width="200" height="200">
				<path
					d="M523.861333 196.053333a320 320 0 0 1 320 320v135.125334a256 256 0 0 1-256 256h-151.722666a256 256 0 0 1-256-256v-135.125334a320 320 0 0 1 320-320z m-13.44 110.08c-54.442667 0-114.346667 11.093333-161.536 35.541334-47.274667 24.405333-86.4 64.981333-86.4 124.032v27.306666c0 72.064 53.333333 134.997333 124.416 134.997334h247.04c71.082667 0 124.416-62.933333 124.416-134.954667v-27.349333c0-59.050667-39.125333-99.626667-86.4-124.032-47.232-24.405333-107.093333-35.584-161.536-35.584z"
					fill="#DDE0E4" p-id="5539"></path>
				<path
					d="M845.610667 188.757333a25.6 25.6 0 0 0-36.224-36.181333l-83.242667 83.242667a344.021333 344.021333 0 0 0-202.282667-65.365334h-23.722666c-74.24 0-143.061333 23.466667-199.381334 63.274667L219.562667 152.576a25.6 25.6 0 1 0-36.181334 36.181333l77.696 77.738667a344.618667 344.618667 0 0 0-106.538666 249.6v135.082667a281.6 281.6 0 0 0 281.6 281.6h151.722666a281.6 281.6 0 0 0 281.6-281.6v-135.125334c0-96.853333-39.850667-184.362667-104.021333-247.125333l80.128-80.170667zM205.738667 516.053333a294.4 294.4 0 0 1 294.4-294.4h23.722666a294.4 294.4 0 0 1 294.4 294.4v135.125334a230.4 230.4 0 0 1-230.4 230.4h-151.722666a230.4 230.4 0 0 1-230.4-230.4v-135.125334z m-94.805334 10.154667a25.6 25.6 0 0 0-51.2 0v165.973333a25.6 25.6 0 1 0 51.2 0v-165.973333z m827.733334-25.6a25.6 25.6 0 0 1 25.6 25.6v165.973333a25.6 25.6 0 1 1-51.2 0v-165.973333a25.6 25.6 0 0 1 25.6-25.6z m-231.509334-7.594667c0 48.810667-35.157333 83.797333-73.216 83.797334h-247.04c-38.058667 0-73.216-34.986667-73.216-83.754667v-27.349333c0-32.426667 20.522667-58.837333 58.666667-78.549334 38.229333-19.712 89.514667-29.866667 138.069333-29.866666 48.512 0 99.84 10.154667 138.026667 29.866666 38.144 19.712 58.709333 46.165333 58.709333 78.506667v27.392z m-73.216 134.997334c71.082667 0 124.416-62.933333 124.416-134.954667v-27.349333c0-59.050667-39.125333-99.626667-86.4-124.032-47.232-24.405333-107.093333-35.584-161.536-35.584-54.442667 0-114.346667 11.178667-161.536 35.584-47.274667 24.405333-86.4 64.981333-86.4 124.032v27.306666c0 72.064 53.333333 134.997333 124.416 134.997334h247.04z m-65.28-149.76a41.002667 41.002667 0 1 0 82.048 0 41.002667 41.002667 0 0 0-82.005333 0z m-157.653333 41.045333a41.002667 41.002667 0 1 1 0-82.005333 41.002667 41.002667 0 0 1 0 82.005333z"
					fill="#66696C" p-id="5540"></path>
			</svg>
			<select name="llm-select no-select" id="model" onchange="cleanhistory()"></select>
			<div class="navbutton no-select" onclick="cleanhistory()"></div>
		</div>
		<div class="chat-container"></div>
		<div class="input-container">
			<textarea id="user-input" class="msginput" name="message"></textarea>
			<div class="submitbutton no-select" onclick="submit()">发送</div>
		</div>
	</div>
</body>

</html>
© 版权声明
THE END
喜欢就支持一下吧
点赞15赞赏 分享
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

取消
昵称表情代码图片

    暂无评论内容