aiBoxRight.vue 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737
  1. <template>
  2. <div class="ai_body" v-loading="loading">
  3. <div class="ai_body_dialog" ref="chatDialog">
  4. <div class="dialog_content" v-for="item in array" :key="item.uid">
  5. <div v-if="item.content" style="margin-left: auto;">
  6. <div class="content content2" v-html="item.content"></div>
  7. <div class="role">
  8. <img src="../../../assets/icon/new/role2.png" />
  9. </div>
  10. </div>
  11. <div style="margin-top:20px;margin-bottom:20px ; margin-right: auto;">
  12. <div class="role">
  13. <img :src="item.fileid
  14. ? item.fileid
  15. : require('../../../assets/icon/new/role1.png')
  16. " />
  17. </div>
  18. <div element-loading-background="#f6f9ff" :style="{
  19. minHeight: item.loading ? '50px' : 'unset',
  20. minWidth: item.loading ? '50px' : 'unset'
  21. }" class="content" v-loading="item.loading">
  22. <span class="vditor-reset" v-html="item.aiContent"></span>
  23. <span class="createTime" v-text="item.createtime"></span>
  24. </div>
  25. <div class="ai_btn_box" v-if="!pan(item.aiContent).length && !item.loading">
  26. <img src="../../../assets/icon/course/pasete.png" @click="onCopy(item.aiContent)" >
  27. </div>
  28. </div>
  29. </div>
  30. </div>
  31. <div class="ai_body_select" v-if="false">
  32. <div class="checkBox" v-if="checkBool">
  33. <div class="task">
  34. <div class="title">选择需要优化的任务:</div>
  35. <div class="content">
  36. <div class="span" @click="addAllTask()">
  37. <div class="check">
  38. <img :src="checkImg" alt="" v-if="checkArray.length !== course.length" />
  39. <img :src="checkIsImg" alt="" v-else />
  40. </div>
  41. <span>全选</span>
  42. </div>
  43. <div class="span" v-for="(item, index) in course" :key="index" @click="addTask(index)">
  44. <div class="check">
  45. <img :src="checkImg" alt="" v-if="checkArray.indexOf(index) === -1" />
  46. <img :src="checkIsImg" alt="" v-else />
  47. </div>
  48. <span>任务{{ index + 1 }}</span>
  49. </div>
  50. </div>
  51. </div>
  52. <div class="part">
  53. <div class="title">选择优化的部分:</div>
  54. <div class="content">
  55. <div class="span" v-for="(item, index) in partArray" :key="index" :class="{ active: part == item.name }"
  56. @click="checkPart(item.name)">
  57. {{ item.name }}
  58. </div>
  59. </div>
  60. </div>
  61. </div>
  62. <span class="check" :class="{ isCheck: checkBool }" v-if="!checkArray.length && !part"
  63. @click="checkBool = !checkBool">选择优化内容</span>
  64. <span class="check" :class="{ isCheck: checkBool }" @click="checkBool = !checkBool" v-else>
  65. <el-tooltip :content="taskName" placement="top" effect="dark">
  66. <!-- content to trigger tooltip here -->
  67. <span>{{ taskName }}</span>
  68. </el-tooltip>
  69. </span>
  70. </div>
  71. <div class="ai_body_input">
  72. <div class="ai_b_i_btnArea">
  73. <span class="clear" @click.stop="clear()">
  74. <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg">
  75. <path fill-rule="evenodd" clip-rule="evenodd"
  76. d="M2.5 3.125C2.5 2.77982 2.77982 2.5 3.125 2.5H16.875C17.2202 2.5 17.5 2.77982 17.5 3.125V8.02715C17.5 8.37233 17.2202 8.65215 16.875 8.65215C16.5298 8.65215 16.25 8.37233 16.25 8.02715V3.75H3.75V16.25H8.125C8.47018 16.25 8.75 16.5298 8.75 16.875C8.75 17.2202 8.47018 17.5 8.125 17.5H3.125C2.77982 17.5 2.5 17.2202 2.5 16.875V3.125Z" />
  77. <path fill-rule="evenodd" clip-rule="evenodd"
  78. d="M5.625 6.1521C5.625 5.80692 5.90482 5.5271 6.25 5.5271H13.125C13.4702 5.5271 13.75 5.80692 13.75 6.1521C13.75 6.49728 13.4702 6.7771 13.125 6.7771H6.25C5.90482 6.7771 5.625 6.49728 5.625 6.1521Z" />
  79. <path fill-rule="evenodd" clip-rule="evenodd"
  80. d="M5.625 9.2771C5.625 8.93192 5.90482 8.6521 6.25 8.6521H9.37496C9.72014 8.6521 9.99996 8.93192 9.99996 9.2771C9.99996 9.62228 9.72014 9.9021 9.37496 9.9021H6.25C5.90482 9.9021 5.625 9.62228 5.625 9.2771Z" />
  81. <path fill-rule="evenodd" clip-rule="evenodd"
  82. d="M12.465 11.507L15.9141 14.9048C16.1279 14.5365 16.25 14.1088 16.25 13.6521C16.25 12.2714 15.1307 11.1521 13.75 11.1521C13.2799 11.1521 12.8406 11.2815 12.465 11.507ZM15.0374 15.7957L11.5873 12.397C11.3726 12.7659 11.25 13.1944 11.25 13.6521C11.25 15.0328 12.3693 16.1521 13.75 16.1521C14.2211 16.1521 14.6613 16.0222 15.0374 15.7957ZM11.0797 11.0192C11.759 10.3303 12.7051 9.9021 13.75 9.9021C15.8211 9.9021 17.5 11.581 17.5 13.6521C17.5 14.6767 17.0882 15.6064 16.4226 16.2827C15.7431 16.9729 14.7961 17.4021 13.75 17.4021C11.6789 17.4021 10 15.7232 10 13.6521C10 12.6263 10.4127 11.6957 11.0797 11.0192Z" />
  83. </svg>
  84. <span>清屏</span>
  85. </span>
  86. <span class="clear" @click.stop="showjList = !showjList" v-if="jArray.length">
  87. <span>查看</span>
  88. </span>
  89. <div style="margin-left: auto;">
  90. <el-switch v-model="continuous"></el-switch>
  91. <span @click.stop="continuous = !continuous">连续对话</span>
  92. </div>
  93. </div>
  94. <div class="ai_b_i_roleListBox" ref="roleListRef" v-if="showRoleList && choseRoleList.length > 0">
  95. <div :class="[
  96. 'ai_b_i_rlb_item',
  97. index == choseRoleItem ? 'ai_b_i_rlb_itemActive' : ''
  98. ]" :ref="`roleItem${index}Ref`" v-for="(item, index) in choseRoleList" :key="item.id"
  99. @mouseover="choseRoleItem = index" @click.stop="choseRole(item)">
  100. <div class="ai_b_i_rlb_itemTop">
  101. <img :src="item.headUrl
  102. ? item.headUrl
  103. : require('../../../assets/icon/new/role1.png')
  104. " alt="" />
  105. <div class="ai_b_i_rlb_i_name">
  106. <span>{{ item.assistantName }}</span>
  107. <span>作者:{{ item.username }}</span>
  108. </div>
  109. </div>
  110. <div class="ai_b_i_rlb_itemBottom">
  111. {{ item.prologue }}
  112. </div>
  113. </div>
  114. </div>
  115. <div class="ai_b_i_jListPanel" @click="showjList = !showjList" v-if="showjList && jArray.length > 0"></div>
  116. <div class="ai_b_i_jListBox" v-if="showjList && jArray.length > 0">
  117. <div class="jlist_box" v-for="(item, index) in jArray" :key="index">
  118. <el-tooltip :content="item.area + ':' + item.value" placement="left" effect="dark" popper-class="text_tooltip2">
  119. <!-- content to trigger tooltip here -->
  120. <span>{{ item.area }}:{{ item.value }}</span>
  121. </el-tooltip>
  122. <div class="cancel" @click="cancelJ(item.string)">×</div>
  123. </div>
  124. </div>
  125. <!-- <div class="ai_b_i_textListBox">
  126. <div class="ai_b_i_tlb_left"></div>
  127. <div class="ai_b_i_tlb_right"></div>
  128. </div> -->
  129. <!-- @input="inputChange" -->
  130. <textarea class="ai_body_input_textarea" @keydown="textareaKeydown" ref="textareaRef" v-model.trim="courseText"
  131. placeholder="在此输入您想了解的内容"></textarea>
  132. <div :class="[
  133. 'c_pub_button_confirm',
  134. courseText ? '' : 'c_pub_button_confirmDisabled'
  135. ]" v-if="!loading" @click="addContent">
  136. 发送
  137. </div>
  138. </div>
  139. <!-- <div class="ai_body_input">
  140. <textarea
  141. style="padding-right: 85px;"
  142. rows="3"
  143. @keyup.enter="addContent"
  144. class="binfo_input binfo_textarea"
  145. cols
  146. v-model.trim="courseText"
  147. placeholder="在此输入您想了解的内容"
  148. ></textarea>
  149. <div
  150. class="c_pub_button_confirm"
  151. v-if="!loading && courseText"
  152. @click="addContent"
  153. >
  154. 发送
  155. </div>
  156. <div class="c_pub_button_confirm" @click="promptTit" v-else>发送</div>
  157. </div> -->
  158. </div>
  159. </template>
  160. <script>
  161. import checkImg from "../../../assets/icon/sourceFile/check.png";
  162. import checkIsImg from "../../../assets/icon/sourceFile/check_is.png";
  163. import { v4 as uuidv4 } from "uuid";
  164. import MarkdownIt from "markdown-it";
  165. import TurndownService from 'turndown';
  166. export default {
  167. props: {
  168. unitJson: {
  169. type: Array
  170. },
  171. courseId: {
  172. type: String,
  173. default: ""
  174. },
  175. quoteArray: {
  176. type: Object,
  177. default: function () {
  178. return {};
  179. }
  180. },
  181. teacherCourseText: {
  182. type: String,
  183. default: ""
  184. },
  185. targetCourseText: {
  186. type: String,
  187. default: ""
  188. },
  189. courseText2: {
  190. type: String,
  191. default: ""
  192. },
  193. targetCourseText2: {
  194. type: String,
  195. default: ""
  196. },
  197. cpote: {
  198. type: Object,
  199. default: function () {
  200. return {};
  201. }
  202. },
  203. },
  204. data() {
  205. return {
  206. array: [],
  207. jArray: [],
  208. courseText: "",
  209. checkImg: checkImg,
  210. checkIsImg: checkIsImg,
  211. userid: this.$route.query.userid,
  212. oid: this.$route.query.oid,
  213. org: this.$route.query.org,
  214. checkArray: [],
  215. course: [{ title: "任务1" }, { title: "任务2" }, { title: "任务3" }],
  216. partArray: [
  217. { name: "全部内容" },
  218. { name: "任务设计" },
  219. { name: "评价设计" }
  220. ],
  221. part: "全部内容",
  222. checkBool: false,
  223. loading: false,
  224. textareaHeight: 50,
  225. publicRoleList: [],
  226. roleList: [],
  227. textList: [
  228. {
  229. title: "项目式学习",
  230. dataList: [
  231. "请给我一些学生开展项目式学习可以使用的主题或问题参考,请说出学生将要解决的问题,以及学生要经历怎样的学习活动。",
  232. "请将一个关于生态保护项目的项目式学习展开描述,你需要描述学生如何解决这个问题,你需要至少写出四个活动,这些活动需要按照前后逻辑关系排列。",
  233. "请对驱动问题为“如何为学校建造一个富有特色的花坛?”的项目式学习进行子问题拆解,至少拆解为5个子问题,并根据子问题对应写出各环节的主要活动。"
  234. ]
  235. },
  236. {
  237. title: "教学评价",
  238. dataList: [
  239. "如果需要给学生的社区服务进行评价,给出评价维度和至少3个等级的表现描述。",
  240. "为6年级学生设计一份关于梧桐山研究报告的评估任务表,并给出参考的报告流程,至少包含8个步骤,并包括地图、图片和至少300个词。",
  241. "创建一个给5年级学生使用的课堂小测试,包含5道多选题,评价学生对于太阳能这个概念的理解。你需要给出题目和正确答案。"
  242. ]
  243. },
  244. {
  245. title: "教学设计",
  246. dataList: [
  247. "如果需要5年级学生感受“移步换景”的景观写作手法,你有什么合适的阅读材料推荐?你需要给出材料名称,以及材料的哪部分内容。",
  248. "设计一个针对8年级学生且关于人类迁徙主题的地理课,并在课程中设计至少1项小组活动。",
  249. "设计一个针对5年级学生的课程,课程综合科学和信息技术领域,解决生物与环境领域的生活问题,你需要给出完整的课程框架和活动。",
  250. "如果3年级的学生不能理解光合作用的实现过程,需要你帮我设计一个支持他们理解的教学活动,需要包含活动的形式、实施材料和清单。",
  251. "请基于贝叶斯定理为8年级学生出三道题目。",
  252. "如果需要八年级学生了解尼罗河流域的文化发展史,你有哪些推荐的网站或参考书籍?"
  253. ]
  254. },
  255. {
  256. title: "班级管理",
  257. dataList: [
  258. "创建一组给一年级学生使用的班级口号,要求大家注意卫生、保护环境,口号需要对仗工整,符合一年级学生的理解水平。",
  259. " 设计一套用于6年级学生的班级管理规章制度,内容需要包括学习、纪律、卫生、思想品德方面。"
  260. ]
  261. },
  262. {
  263. title: "课堂组织",
  264. dataList: [
  265. "请为“制作垃圾分类宣传单”的小组活动设计小组分工表,每个小组的成员为4-6人。",
  266. "请给5年级“校园植物图鉴”社团课程设计一份小组合作公约,需包含小组成员信息、小组项目目标、填写日期,总长度不超过300字,并且提供至少3处学生自行填写的部分。",
  267. "请用苏格拉底提问的方式,引导5年级学生拆解驱动问题:如何解决教室黑板反光的问题?其中需包含对于反光原因的分析与实验探究。"
  268. ]
  269. },
  270. {
  271. title: "教师发展",
  272. dataList: [
  273. "教师需要理解项目式学习的理论基础和基础概念,你需要生成一份阅读清单,要求内容为中文书籍或文献。",
  274. "设计一个教师进行个人学期总结的框架,需要体现在教学、教研、个人学习方面的进步。"
  275. ]
  276. },
  277. {
  278. title: "代码分析",
  279. dataList: [
  280. "这段代码实现了什么效果?",
  281. "请描述这段代码。",
  282. "根据这段代码,给我一些修改意见。"
  283. ]
  284. }
  285. ],
  286. showTextList: false,
  287. showRoleList: false,
  288. choseRoleItem: 0,
  289. choseTextItem: 0,
  290. continuous: true,
  291. showjList: false
  292. };
  293. },
  294. watch: {
  295. unitJson: {
  296. immediate: true,
  297. deep: true,
  298. handler(newValue, oldValue) {
  299. this.course = this.unitJson[0].chapterInfo[0].taskJson;
  300. this.jArray = this.getString();
  301. if (!this.jArray.length) {
  302. this.showjList = false
  303. }
  304. }
  305. },
  306. courseId: {
  307. immediate: true,
  308. deep: true,
  309. handler(newValue, oldValue) {
  310. if (newValue) {
  311. this.getChatList().then(_ => {
  312. this.$nextTick(() => {
  313. console.log(this.$refs.chatDialog.scrollHeight);
  314. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  315. });
  316. });
  317. }
  318. }
  319. },
  320. quoteArray: {
  321. immediate: true,
  322. deep: true,
  323. handler(newValue, oldValue) {
  324. this.jArray = this.getString();
  325. if (!this.jArray.length) {
  326. this.showjList = false
  327. }
  328. }
  329. },
  330. teacherCourseText: {
  331. immediate: true,
  332. deep: true,
  333. handler(newValue, oldValue) {
  334. this.jArray = this.getString();
  335. if (!this.jArray.length) {
  336. this.showjList = false
  337. }
  338. }
  339. },
  340. targetCourseText: {
  341. immediate: true,
  342. deep: true,
  343. handler(newValue, oldValue) {
  344. this.jArray = this.getString();
  345. if (!this.jArray.length) {
  346. this.showjList = false
  347. }
  348. }
  349. },
  350. courseText2: {
  351. immediate: true,
  352. deep: true,
  353. handler(newValue, oldValue) {
  354. this.jArray = this.getString();
  355. if (!this.jArray.length) {
  356. this.showjList = false
  357. }
  358. }
  359. },
  360. targetCourseText2: {
  361. immediate: true,
  362. deep: true,
  363. handler(newValue, oldValue) {
  364. this.jArray = this.getString();
  365. if (!this.jArray.length) {
  366. this.showjList = false
  367. }
  368. }
  369. },
  370. cpote: {
  371. immediate: true,
  372. deep: true,
  373. handler(newValue, oldValue) {
  374. this.jArray = this.getString();
  375. if (!this.jArray.length) {
  376. this.showjList = false
  377. }
  378. }
  379. },
  380. },
  381. methods: {
  382. cancelJ(string) {
  383. this.$emit('setIsQuote', string)
  384. },
  385. getString() {
  386. let quoteArray = []
  387. let array = Object.keys(this.quoteArray)
  388. array.forEach(item => {
  389. if (!this.quoteArray[item]) {
  390. return
  391. }
  392. let string = item
  393. let area = ''
  394. let text = ''
  395. if (string == 'aiteacherTextDetail') {
  396. area = '已有教案摘要'
  397. text = this.teacherCourseText
  398. } else if (string == 'aitargetTextDetail') {
  399. area = '课程目标'
  400. text = this.targetCourseText
  401. } else if (string == 'aiDetail') {
  402. area = '课程简要描述'
  403. text = this.courseText2
  404. } else if (string == 'aitargetTextDetail2') {
  405. area = '课程目标'
  406. text = this.targetCourseText2
  407. } else if (string.indexOf('aiDetail1') != -1) {
  408. let index = parseInt(string.split('aiDetail1-')[1])
  409. area = `任务${index + 1}课程大纲`
  410. text = this.unitJson[0].chapterInfo[0].taskJson[index].taskDetail2
  411. } else if (string == 'aiCpote1') {
  412. area = '概念群'
  413. text = this.cpote.cpote1
  414. } else if (string == 'aiCpote2') {
  415. area = '问题链'
  416. text = this.cpote.cpote2
  417. } else if (string == 'aiCpote3') {
  418. area = '目标层'
  419. text = this.cpote.cpote3
  420. } else if (string == 'aiCpote4') {
  421. area = '任务簇'
  422. text = this.cpote.cpote4
  423. } else if (string.indexOf('teacherDetail2') != -1) {
  424. let index = parseInt(string.split('teacherDetail2-')[1])
  425. area = `任务${index + 1}教案`
  426. text = this.unitJson[0].chapterInfo[0].taskJson[index].taskDetail3
  427. } else if (string.indexOf('aiDetail2') != -1) {
  428. let index = parseInt(string.split('aiDetail2-')[1])
  429. area = `任务${index + 1}任务描述`
  430. text = this.unitJson[0].chapterInfo[0].taskJson[index].taskDetail
  431. } else if (string.indexOf('aiDetail3') != -1) {
  432. let index = parseInt(string.split('-')[1])
  433. let index2 = parseInt(string.split('-')[2])
  434. area = `任务${index + 1}工具${index2 + 1}的工具描述`
  435. text = this.unitJson[0].chapterInfo[0].taskJson[index].toolChoose[index2].toolDetail
  436. }
  437. quoteArray.push({
  438. string,
  439. area,
  440. value: text
  441. })
  442. })
  443. return quoteArray
  444. },
  445. promptTit() {
  446. if (!this.loading && !this.courseText) {
  447. this.$message({
  448. message: "请输入您想要了解的内容",
  449. type: "warning"
  450. });
  451. } else {
  452. this.$message({
  453. message: "请回答完毕后再次发送",
  454. type: "warning"
  455. });
  456. }
  457. },
  458. addContent() {
  459. let message = this.courseText;
  460. if (this.courseText) {
  461. let msg = ``;
  462. if (this.checkArray.length) {
  463. msg = `
  464. ATTENTION: Use '##' to SPLIT SECTIONS, not '#'.Output format carefully referenced "Format example".`
  465. let task = [];
  466. if (this.part == "全部内容") {
  467. for (var i = 0; i < this.checkArray.length; i++) {
  468. let _index = this.checkArray[i];
  469. task.push(this.course[_index]);
  470. }
  471. } else if (this.part == "任务设计") {
  472. for (var i = 0; i < this.checkArray.length; i++) {
  473. let _index = this.checkArray[i];
  474. task.push(this.course[_index]);
  475. }
  476. } else if (this.part == "评价设计") {
  477. for (var i = 0; i < this.checkArray.length; i++) {
  478. let _index = this.checkArray[i];
  479. task.push(this.course[_index].eList);
  480. }
  481. }
  482. msg += `
  483. ## 修改内容
  484. ${JSON.stringify(task)}
  485. `;
  486. msg += `
  487. ## 要求
  488. ${this.courseText}
  489. `;
  490. message = msg;
  491. }
  492. if (this.jArray.length) {
  493. let task = [];
  494. for (var i = 0; i < this.jArray.length; i++) {
  495. let _index = this.jArray[i];
  496. task.push(`${_index.area}: ${_index.value.replaceAll('\n', '')}`);
  497. }
  498. msg += `
  499. ## 参考资料
  500. ${task.join('\n')}
  501. `;
  502. msg += `
  503. ## 要求
  504. 根据<参考资料>中的内容实现以下要求:${this.courseText}
  505. `;
  506. message = msg;
  507. }
  508. // 这里处理@的角色
  509. let _atRoleList = [];
  510. let _roleList = [...this.roleList, ...this.publicRoleList];
  511. _roleList.forEach(i => {
  512. if (message.indexOf(`@${i.assistantName}`) != -1) {
  513. _atRoleList.push(i);
  514. }
  515. });
  516. if (_atRoleList.length > 0) {
  517. //有@角色
  518. let _replaceText = message;
  519. let _htmlText = message;
  520. _atRoleList.forEach(_i => {
  521. _replaceText = _replaceText.replaceAll(`@${_i.assistantName}`, ``);
  522. _htmlText = _htmlText.replaceAll(
  523. `@${_i.assistantName}`,
  524. `<span class='aite-name'>@${_i.assistantName}</span>`
  525. );
  526. });
  527. _atRoleList.forEach((_item, _index) => {
  528. const _uid = uuidv4();
  529. if (_index == 0) {
  530. this.array.push({
  531. loading: true,
  532. role: "user",
  533. content: _htmlText,
  534. uid: _uid,
  535. AI: "AI",
  536. aiContent: "",
  537. oldContent: "",
  538. isShowSynchronization: false,
  539. filename: _item.assistantName,
  540. index: this.array.length,
  541. is_mind_map: false,
  542. fileid: _item.headUrl,
  543. createtime: new Date().toLocaleString().replaceAll("/", "-")
  544. });
  545. } else {
  546. this.array.push({
  547. loading: true,
  548. role: "user",
  549. content: "",
  550. uid: _uid,
  551. AI: "AI",
  552. aiContent: "",
  553. oldContent: "",
  554. isShowSynchronization: false,
  555. filename: _item.assistantName,
  556. index: this.array.length,
  557. is_mind_map: false,
  558. fileid: _item.headUrl,
  559. createtime: new Date().toLocaleString().replaceAll("/", "-")
  560. });
  561. }
  562. this.$nextTick(() => {
  563. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  564. });
  565. let params = {
  566. assistant_id: _item.assistant_id,
  567. userId: this.userid,
  568. message: _replaceText,
  569. session_name: `${this.courseId}-addCourseA`,
  570. uid: _uid,
  571. file_ids: [],
  572. model: "gpt-4o-2024-08-06"
  573. };
  574. var OpenCC = require("opencc-js");
  575. let converter = OpenCC.Converter({
  576. from: "cn",
  577. to: "hk"
  578. });
  579. this.ajax
  580. .post("https://gpt4.cocorobo.cn/ai_agent_park_chat_new", params)
  581. .then(res => {
  582. if (
  583. converter(res.data.FunctionResponse.result) ==
  584. converter("发送成功")
  585. ) {
  586. } else {
  587. this.$message.warning(res.data.FunctionResponse.result);
  588. }
  589. })
  590. .catch(err => {
  591. console.log(err);
  592. });
  593. this.getAtAuContent(_uid);
  594. });
  595. this.courseText = "";
  596. } else {
  597. let _uuid = uuidv4();
  598. this.array.push({
  599. role: "user",
  600. content: `${this.courseText}`,
  601. uid: _uuid,
  602. AI: "AI",
  603. aiContent: "",
  604. oldContent: "",
  605. isShowSynchronization: false,
  606. filename: "",
  607. index: this.array.length,
  608. is_mind_map: false,
  609. createtime: new Date().toLocaleString().replaceAll("/", "-"),
  610. loading: true
  611. });
  612. let history = [];
  613. if (this.continuous) {
  614. this.array.forEach(i => {
  615. if (i.content) history.push({ role: "user", content: i.content });
  616. if (i.aiContent)
  617. history.push({ role: "assistant", content: i.aiContent });
  618. });
  619. }
  620. history.push({ role: "user", content: message });
  621. this.$nextTick(() => {
  622. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  623. });
  624. let params = JSON.stringify({
  625. // model: "gpt-3.5-turbo",
  626. // model: "gpt-4o-2024-08-06",
  627. model: "qwen-plus",
  628. temperature: 0,
  629. max_tokens: 4096,
  630. top_p: 1,
  631. frequency_penalty: 0,
  632. presence_penalty: 0,
  633. messages: history,
  634. uid: _uuid,
  635. mind_map_question: ""
  636. });
  637. this.courseText = "";
  638. this.ajax
  639. .post("https://gpt4.cocorobo.cn/chat", params)
  640. .then(res => {
  641. if (res.data.FunctionResponse.result == "发送成功") {
  642. } else {
  643. this.$message.warning(res.data.FunctionResponse.result);
  644. }
  645. })
  646. .catch(e => {
  647. console.log(e);
  648. });
  649. this.getAiContent(_uuid);
  650. }
  651. }
  652. },
  653. getAiContent(_uid) {
  654. let _source = new EventSource(`https://gpt4.cocorobo.cn/stream/${_uid}`); //http://gpt4.cocorobo.cn:8011/stream/ https://gpt4.cocorobo.cn/stream/
  655. let _allText = "";
  656. let _mdText = "";
  657. const md = new MarkdownIt();
  658. _source.onmessage = _e => {
  659. if (_e.data.replace("'", "").replace("'", "") == "[DONE]") {
  660. //对话已经完成
  661. _mdText = _mdText.replace("_", "");
  662. _source.close();
  663. this.$nextTick(() => {
  664. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  665. });
  666. this.array.find(i => i.uid == _uid).aiContent = _mdText;
  667. this.array.find(i => i.uid == _uid).isalltext = true;
  668. this.array.find(i => i.uid == _uid).isShowSynchronization = true;
  669. this.array.find(i => i.uid == _uid).loading = false;
  670. // 这里保存对话
  671. if (this.courseId) {
  672. this.insertChat(_uid);
  673. }
  674. return;
  675. } else {
  676. //对话还在继续
  677. let _text = "";
  678. _text = _e.data.replaceAll("'", "");
  679. if (_allText == "") {
  680. _allText = _text.replace(/^\n+/, ""); //去掉回复消息中偶尔开头就存在的连续换行符
  681. } else {
  682. _allText += _text;
  683. }
  684. _mdText = _allText + "_";
  685. _mdText = _mdText.replace(/\\n/g, "\n");
  686. _mdText = _mdText.replace(/\\/g, "");
  687. if (_allText.split("```").length % 2 == 0) _mdText += "\n```\n";
  688. //转化返回的回复流数据
  689. _mdText = md.render(_mdText);
  690. this.array.find(i => i.uid == _uid).aiContent = _mdText;
  691. this.array.find(i => i.uid == _uid).loading = false;
  692. this.$nextTick(() => {
  693. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  694. });
  695. // 处理流数据
  696. }
  697. };
  698. },
  699. getAtAuContent(_uid) {
  700. let _source = new EventSource(
  701. `https://gpt4.cocorobo.cn/question/${_uid}`
  702. ); //http://gpt4.cocorobo.cn:8011/question/ https://gpt4.cocorobo.cn/question/
  703. let _allText = "";
  704. let _mdText = "";
  705. const md = new MarkdownIt();
  706. _source.onmessage = _e => {
  707. let _eData = JSON.parse(_e.data);
  708. if (_eData.content.replace("'", "").replace("'", "") == "[DONE]") {
  709. let _result = [];
  710. if ("result" in _eData) {
  711. _result = _eData.result;
  712. for (let i = 0; i < _result.length; i++) {
  713. _mdText = _mdText.replace(_result[i].text, _result[i].fileName);
  714. }
  715. }
  716. _mdText = _mdText.replace("_", "");
  717. this.array.find(i => i.uid == _uid).aiContent = _mdText;
  718. this.array.find(i => i.uid == _uid).isalltext = true;
  719. this.array.find(i => i.uid == _uid).isShowSynchronization = true;
  720. this.array.find(i => i.uid == _uid).loading = false;
  721. this.$nextTick(() => {
  722. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  723. });
  724. this.insertChat(_uid);
  725. } else {
  726. let _text = _eData.content.replace("'", "").replace("'", "");
  727. if (_allText == "") {
  728. _allText = _text.replace(/^\n+/, ""); //去掉回复消息中偶尔开头就存在的连续换行符
  729. } else {
  730. _allText += _text;
  731. }
  732. _mdText = _allText + "_";
  733. _mdText = _mdText.replace(/\\n/g, "\n");
  734. _mdText = _mdText.replace(/\\/g, "");
  735. if (_allText.split("```").length % 2 == 0) _mdText += "\n```\n";
  736. //转化返回的回复流数据
  737. _mdText = md.render(_mdText);
  738. this.array.find(i => i.uid == _uid).aiContent = _mdText;
  739. this.array.find(i => i.uid == _uid).loading = false;
  740. this.$nextTick(() => {
  741. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  742. });
  743. // 处理流数据
  744. }
  745. };
  746. },
  747. //保存消息
  748. insertChat(_uid) {
  749. let _data = this.array.find(i => i.uid == _uid);
  750. if (!_data) return;
  751. let params = {
  752. userId: this.userid,
  753. userName: "qgt",
  754. groupId: "602def61-005d-11ee-91d8-005056b8q12w",
  755. answer: _data.aiContent,
  756. problem: _data.content,
  757. file_id: _data.fileid ? _data.fileid : "",
  758. alltext: _data.aiContent,
  759. type: "chat",
  760. filename: _data.filename,
  761. session_name: `${this.courseId}-addCourseA` //这是对话记录位置
  762. };
  763. this.ajax
  764. .post("https://gpt4.cocorobo.cn/insert_chat", params)
  765. .then(res => { });
  766. },
  767. // 获取对应的聊天记录
  768. getChatList() {
  769. return new Promise((resolve, reject) => {
  770. if (this.loading) return this.$message.info("请稍等...");
  771. this.array = [];
  772. this.loading = true;
  773. let params = {
  774. userid: this.userid,
  775. groupid: "602def61-005d-11ee-91d8-005056b8q12w",
  776. // session_name:``
  777. session_name: `${this.courseId}-addCourseA`
  778. };
  779. this.ajax
  780. .post("https://gpt4.cocorobo.cn/get_agent_park_chat", params)
  781. .then(res => {
  782. let _data = JSON.parse(res.data.FunctionResponse);
  783. if (_data.length > 0) {
  784. let _chatList = [];
  785. for (let i = 0; i < _data.length; i++) {
  786. _chatList.push({
  787. loading: false,
  788. role: "user",
  789. content: _data[i].problem,
  790. uid: _data[i].id,
  791. AI: "AI",
  792. aiContent: _data[i].answer,
  793. oldContent: _data[i].answer,
  794. isShowSynchronization: false,
  795. filename: _data[i].filename,
  796. index: i,
  797. is_mind_map: false,
  798. fileid: _data[i].fileid,
  799. createtime: _data[i].createtime
  800. });
  801. }
  802. this.array = _chatList;
  803. this.loading = false;
  804. } else {
  805. let _uid = uuidv4();
  806. let _chatList = [];
  807. _chatList.push({
  808. loading: false,
  809. role: "",
  810. content: "",
  811. uid: _uid,
  812. AI: "AI",
  813. aiContent:
  814. "您好,我是您的创课助手小可,在创课中遇到什么问题,都可以与我对话~,我会尽量帮助您的",
  815. oldContent:
  816. "您好,我是您的创课助手小可,在创课中遇到什么问题,都可以与我对话~,我会尽量帮助您的",
  817. isShowSynchronization: false,
  818. filename: "",
  819. index: 0,
  820. is_mind_map: false,
  821. fileid: ""
  822. });
  823. this.array = _chatList;
  824. if (this.courseId) {
  825. this.insertChat(_uid);
  826. }
  827. //没有对话记录
  828. this.loading = false;
  829. }
  830. resolve();
  831. })
  832. .catch(err => {
  833. console.log(err);
  834. this.$message.error("获取对话记录失败");
  835. this.loading = false;
  836. resolve();
  837. });
  838. });
  839. },
  840. addTask(index) {
  841. if (this.checkArray.indexOf(index) !== -1) {
  842. this.checkArray.splice(this.checkArray.indexOf(index), 1);
  843. } else {
  844. this.checkArray.push(index);
  845. }
  846. console.log(index);
  847. },
  848. addAllTask() {
  849. if (this.checkArray.length === this.course.length) {
  850. this.checkArray = [];
  851. } else {
  852. this.checkArray = [];
  853. this.course.forEach((item, index) => {
  854. this.checkArray.push(index);
  855. });
  856. }
  857. },
  858. checkPart(name) {
  859. this.part = name;
  860. },
  861. inputChange() {
  862. if (this.courseText.at(-1) == "@") {
  863. this.showRoleList = true;
  864. }
  865. if (this.courseText.at(-1) == "/") {
  866. console.log("哇卡ka2");
  867. }
  868. this.$nextTick(() => {
  869. this.$refs.textareaRef.style.height = "35px";
  870. this.$refs.textareaRef.style.height =
  871. this.$refs.textareaRef.scrollHeight + "px";
  872. this.textareaHeight = this.$refs.textareaRef.style.height;
  873. });
  874. },
  875. textareaKeydown(_e) {
  876. if (this.showRoleList && this.choseRoleList.length > 0) {
  877. console.log(_e.keyCode);
  878. switch (_e.keyCode) {
  879. case 38: //小键盘上
  880. _e.preventDefault();
  881. if (this.choseRoleItem == 0) return;
  882. this.choseRoleItem--;
  883. // 修改滚动条高度
  884. this.$refs.roleListRef.scrollTo({
  885. top:
  886. this.$refs[`roleItem${this.choseRoleItem}Ref`][0].offsetTop -
  887. 10,
  888. behavior: "smooth"
  889. });
  890. // this.$refs.roleListRef.scrollTop = this.choseRoleItem * 107;
  891. break;
  892. case 40: //小键盘下
  893. _e.preventDefault();
  894. if (this.choseRoleItem == this.choseRoleList.length - 1) return;
  895. this.choseRoleItem++;
  896. this.$refs.roleListRef.scrollTo({
  897. top:
  898. this.$refs[`roleItem${this.choseRoleItem}Ref`][0].offsetTop -
  899. 10,
  900. behavior: "smooth"
  901. });
  902. // this.$refs.roleListRef.scrollTop = this.choseRoleItem * 107;
  903. break;
  904. case 13: //回车
  905. _e.preventDefault();
  906. this.choseRole(this.choseRoleList[this.choseRoleItem]);
  907. break;
  908. }
  909. } else if (_e.key === "Enter") {
  910. _e.preventDefault();
  911. if (_e.shiftKey) {
  912. this.courseText += "\n";
  913. } else {
  914. this.addContent();
  915. }
  916. }
  917. },
  918. clear() {
  919. this.$confirm("确定清空聊天记录吗?", "提示", {
  920. confirmButtonText: "确定",
  921. cancelButtonText: "取消",
  922. type: "warning"
  923. })
  924. .then(_ => {
  925. this.loading = true;
  926. let params = {
  927. user_id: this.userid,
  928. id: "602def61-005d-11ee-91d8-005056b8q12w",
  929. session_name: `${this.courseId}-addCourseA`
  930. };
  931. this.ajax
  932. .post("https://gpt4.cocorobo.cn/delete_park_session", params)
  933. .then(res => {
  934. this.array = [];
  935. this.$message.success("清除聊天记录成功");
  936. this.loading = false;
  937. })
  938. .catch(err => {
  939. this.loading = false;
  940. this.$message.error("清除聊天记录失败");
  941. });
  942. })
  943. .catch(_ => { });
  944. },
  945. getRoleList() {
  946. this.roleList = [];
  947. let params = {
  948. userId: this.userid
  949. };
  950. this.ajax
  951. .post("https://gpt4.cocorobo.cn/get_ai_agent_assistant_list", params)
  952. .then(res => {
  953. let _data = res.data.FunctionResponse.result;
  954. if (_data) {
  955. this.roleList = JSON.parse(_data);
  956. }
  957. })
  958. .catch(e => {
  959. console.log(e);
  960. // this.$message.error("获取角色列表失败");
  961. this.roleList = [];
  962. });
  963. },
  964. getPublicRoleList() {
  965. this.publicRoleList = [];
  966. let params = {
  967. userId: this.userid,
  968. organizeid: this.oid
  969. };
  970. this.ajax
  971. .post(
  972. "https://gpt4.cocorobo.cn/get_ai_agent_assistant_share_list",
  973. params
  974. )
  975. .then(res => {
  976. let _data = res.data.FunctionResponse.result;
  977. if (_data) {
  978. this.publicRoleList = JSON.parse(_data);
  979. }
  980. })
  981. .catch(e => {
  982. this.publicRoleList = [];
  983. console.log(e);
  984. // console.log("获取公共角色失败", e);
  985. });
  986. },
  987. choseRole(item) {
  988. let _lastAtIndex = this.courseText.lastIndexOf("@");
  989. this.courseText = `${this.courseText.slice(0, _lastAtIndex)}@${item.assistantName
  990. } `;
  991. this.$refs.textareaRef.focus();
  992. this.showRoleList = false;
  993. },
  994. onCopy(content) {
  995. const turndownService = new TurndownService();
  996. // 添加自定义规则来处理表格
  997. turndownService.addRule('table', {
  998. filter: 'table',
  999. replacement: (content, node) => {
  1000. const rows = node.querySelectorAll('tr');
  1001. let markdown = '';
  1002. rows.forEach((row) => {
  1003. const cells = row.querySelectorAll('th, td');
  1004. const rowMarkdown = Array.from(cells).map(cell => cell.textContent).join(' | ');
  1005. markdown += `| ${rowMarkdown} |\n`;
  1006. if(cells && cells.length && cells[0].tagName == 'TH'){
  1007. let a = Array.from(cells).map(cell => '').join(' --- |')
  1008. markdown += `| --- |${a}\n`;
  1009. }
  1010. });
  1011. // 添加分隔行
  1012. // markdown = markdown.replace(/^/, '| --- |\n');
  1013. return markdown;
  1014. },
  1015. });
  1016. // 创建临时textarea元素
  1017. const tempInput = document.createElement("textarea");
  1018. tempInput.value = turndownService.turndown(content); // 设置要复制的内容
  1019. // 隐藏元素
  1020. tempInput.style.position = "absolute";
  1021. tempInput.style.left = "-9999px";
  1022. // 将元素添加到DOM中
  1023. document.body.appendChild(tempInput);
  1024. // 选中元素内容
  1025. tempInput.select();
  1026. // 执行复制操作
  1027. document.execCommand("copy");
  1028. // 移除临时元素
  1029. document.body.removeChild(tempInput);
  1030. this.$message({
  1031. message: "复制成功",
  1032. type: "success"
  1033. });
  1034. },
  1035. },
  1036. computed: {
  1037. pan() {
  1038. return content => {
  1039. try {
  1040. return JSON.parse(content);
  1041. } catch (error) {
  1042. return [];
  1043. }
  1044. };
  1045. },
  1046. courseTextLength() {
  1047. return this.courseText.length;
  1048. },
  1049. taskName() {
  1050. let task = "";
  1051. if (this.checkArray.length) {
  1052. task = "任务";
  1053. this.checkArray = this.checkArray.sort((a, b) => a - b);
  1054. let a = JSON.parse(JSON.stringify(this.checkArray));
  1055. for (let index = 0; index < a.length; index++) {
  1056. a[index]++;
  1057. }
  1058. task += a.join("/");
  1059. }
  1060. return task + " " + this.part;
  1061. },
  1062. choseRoleList() {
  1063. let result = [...this.roleList, ...this.publicRoleList];
  1064. const _index = this.courseText.lastIndexOf("@");
  1065. if (_index !== -1) {
  1066. let roleName = this.courseText.substring(_index + 1);
  1067. result = result.filter(i => i.assistantName.indexOf(roleName) != -1);
  1068. } else {
  1069. return [];
  1070. }
  1071. this.choseRoleItem = 0;
  1072. return result;
  1073. }
  1074. },
  1075. mounted() {
  1076. this.getChatList().then(_ => {
  1077. this.$nextTick(() => {
  1078. console.log(this.$refs.chatDialog.scrollHeight);
  1079. this.$refs.chatDialog.scrollTop = this.$refs.chatDialog.scrollHeight;
  1080. });
  1081. });
  1082. this.getRoleList();
  1083. this.getPublicRoleList();
  1084. this.jArray = this.getString();
  1085. }
  1086. };
  1087. </script>
  1088. <style scoped>
  1089. .ai_body {
  1090. height: 100%;
  1091. width: calc(100% - 20px);
  1092. margin: 0 auto;
  1093. display: flex;
  1094. flex-direction: column;
  1095. }
  1096. .binfo_input {
  1097. width: 100%;
  1098. margin: 0;
  1099. padding: 12px 14px;
  1100. display: block;
  1101. min-width: 0;
  1102. outline: none;
  1103. box-sizing: border-box;
  1104. background: none;
  1105. border: none;
  1106. border-radius: 4px;
  1107. background: #fff;
  1108. font-size: 14px;
  1109. resize: none;
  1110. font-family: "Microsoft YaHei";
  1111. min-height: 48px;
  1112. /* border: 1px solid #3682fc00; */
  1113. border: 1.5px solid #cad1dc;
  1114. }
  1115. .binfo_textarea {
  1116. border: 1.5px solid #cad1dc;
  1117. font-size: 14px;
  1118. resize: none;
  1119. /* background: #f6f6f6; */
  1120. font-family: "Microsoft YaHei";
  1121. }
  1122. .binfo_textarea::-webkit-scrollbar {
  1123. /*滚动条整体样式*/
  1124. width: 6px;
  1125. /*高宽分别对应横竖滚动条的尺寸*/
  1126. height: 6px;
  1127. }
  1128. /*定义滚动条轨道 内阴影+圆角*/
  1129. .binfo_textarea::-webkit-scrollbar-track {
  1130. border-radius: 10px;
  1131. background-color: rgba(0, 0, 0, 0.1);
  1132. }
  1133. /*定义滑块 内阴影+圆角*/
  1134. .binfo_textarea::-webkit-scrollbar-thumb {
  1135. border-radius: 10px;
  1136. -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3);
  1137. background-color: rgba(0, 0, 0, 0.1);
  1138. }
  1139. .binfo_input:focus-visible {
  1140. border: 1.5px solid #3681fc !important;
  1141. }
  1142. .ai_body_input {
  1143. position: relative;
  1144. display: flex;
  1145. margin-top: auto;
  1146. width: 100%;
  1147. /* height: auto; */
  1148. /* max-height: 80vh; */
  1149. justify-content: space-between;
  1150. align-items: flex-end;
  1151. border-radius: 10px;
  1152. border: 1.5px solid #3681fc !important;
  1153. /* padding: 10px;
  1154. padding-top: 20px; */
  1155. /* overflow: auto; */
  1156. }
  1157. .ai_b_i_btnArea {
  1158. width: calc(100% - 10px);
  1159. position: absolute;
  1160. bottom: calc(100% + 5px);
  1161. height: 30px;
  1162. display: flex;
  1163. /* justify-content: space-between; */
  1164. }
  1165. .ai_b_i_btnArea>div {
  1166. display: flex;
  1167. align-items: center;
  1168. }
  1169. .ai_b_i_btnArea>div>span {
  1170. margin-left: 5px;
  1171. cursor: pointer;
  1172. }
  1173. .ai_b_i_btnArea>.clear+.clear {
  1174. margin-left: 10px;
  1175. }
  1176. .ai_b_i_btnArea>.clear {
  1177. box-sizing: border-box;
  1178. padding: 5px 10px;
  1179. box-sizing: border-box;
  1180. cursor: pointer;
  1181. border: solid 1px #3781fb;
  1182. border-radius: 15px;
  1183. display: flex;
  1184. font-size: 15px;
  1185. align-items: center;
  1186. justify-content: center;
  1187. background-color: #fff;
  1188. }
  1189. .ai_b_i_btnArea>.clear>svg {
  1190. width: 17px;
  1191. height: 17px;
  1192. margin-right: 5px;
  1193. fill: black;
  1194. }
  1195. .ai_b_i_btnArea>.clear:hover {
  1196. background-color: #ebf4fe;
  1197. color: #409eff;
  1198. }
  1199. .ai_b_i_btnArea>.clear:hover>svg {
  1200. fill: #409eff;
  1201. }
  1202. .ai_b_i_textListBox {
  1203. width: 100%;
  1204. height: 300px;
  1205. background-color: #fff;
  1206. position: absolute;
  1207. bottom: calc(100% + 5px);
  1208. box-sizing: border-box;
  1209. padding: 10px;
  1210. overflow: auto;
  1211. border-radius: 8px;
  1212. border: 1px solid #e7e7e7;
  1213. box-shadow: 0 4px 10px 0 rgba(29, 57, 131, 0.08),
  1214. 1px 1px 20px 4px rgba(29, 57, 131, 0.05);
  1215. display: flex;
  1216. }
  1217. .ai_b_i_tlb_left {
  1218. width: 60px;
  1219. height: 100%;
  1220. background-color: red;
  1221. }
  1222. .ai_b_i_tlb_right {
  1223. flex: 1;
  1224. height: 100%;
  1225. background-color: yellow;
  1226. }
  1227. .ai_b_i_roleListBox {
  1228. width: 100%;
  1229. height: 300px;
  1230. background-color: #fff;
  1231. position: absolute;
  1232. bottom: calc(100% + 5px);
  1233. box-sizing: border-box;
  1234. padding: 10px;
  1235. overflow: auto;
  1236. border-radius: 8px;
  1237. border: 1px solid #e7e7e7;
  1238. box-shadow: 0 4px 10px 0 rgba(29, 57, 131, 0.08),
  1239. 1px 1px 20px 4px rgba(29, 57, 131, 0.05);
  1240. }
  1241. .ai_b_i_rlb_item {
  1242. width: calc(100% - 20px);
  1243. height: auto;
  1244. padding: 10px;
  1245. background-color: #f3f7fd;
  1246. margin-bottom: 20px;
  1247. border-radius: 8px;
  1248. display: flex;
  1249. flex-direction: column;
  1250. justify-content: center;
  1251. transition: 0.3s;
  1252. cursor: pointer;
  1253. }
  1254. .ai_b_i_rlb_itemActive {
  1255. background-color: #d1d5db !important;
  1256. }
  1257. /* .ai_b_i_rlb_item:hover{
  1258. background-color: #d1d5db;
  1259. } */
  1260. .ai_b_i_rlb_itemTop {
  1261. display: flex;
  1262. }
  1263. .ai_b_i_rlb_itemTop>img {
  1264. width: 40px;
  1265. height: 40px;
  1266. margin-right: 10px;
  1267. border-radius: 100%;
  1268. overflow: hidden;
  1269. }
  1270. .ai_b_i_rlb_itemTop>div {
  1271. display: flex;
  1272. flex-direction: column;
  1273. }
  1274. .ai_b_i_rlb_itemTop>div>span {
  1275. font-size: 16px;
  1276. font-weight: bold;
  1277. }
  1278. .ai_b_i_rlb_itemTop>div>span:last-child {
  1279. font-size: 14px;
  1280. color: #999;
  1281. }
  1282. .ai_b_i_rlb_itemBottom {
  1283. margin-top: 10px;
  1284. display: flex;
  1285. }
  1286. .ai_b_i_rlb_itemBottom>span {
  1287. width: 60px;
  1288. height: 30px;
  1289. border-radius: 15px;
  1290. display: flex;
  1291. }
  1292. .ai_body_input_textarea {
  1293. flex: 1;
  1294. margin: 10px 5px 10px 5px;
  1295. min-height: 35px;
  1296. height: 35px;
  1297. max-height: 60vh;
  1298. border: none;
  1299. outline: none;
  1300. resize: none;
  1301. font-size: 18px;
  1302. overflow: auto;
  1303. padding-right: 100px;
  1304. }
  1305. .ai_body_input_textarea::-webkit-input-placeholder {
  1306. font-size: 16px;
  1307. /* 修改placeholder字体大小 */
  1308. color: grey;
  1309. /* 修改placeholder文字颜色 */
  1310. }
  1311. .ai_body_input_textarea::-moz-placeholder {
  1312. font-size: 16px;
  1313. /* 修改placeholder字体大小 */
  1314. color: grey;
  1315. /* 修改placeholder文字颜色 */
  1316. opacity: 1;
  1317. /* 修复Firefox的透明度问题 */
  1318. }
  1319. .ai_body_input_textarea::-moz-placeholder {
  1320. font-size: 16px;
  1321. /* 修改placeholder字体大小 */
  1322. color: grey;
  1323. /* 修改placeholder文字颜色 */
  1324. opacity: 1;
  1325. /* 修复Firefox的透明度问题 */
  1326. }
  1327. .ai_body_input_textarea::-ms-input-placeholder {
  1328. font-size: 16px;
  1329. /* 修改placeholder字体大小 */
  1330. color: grey;
  1331. /* 修改placeholder文字颜色 */
  1332. }
  1333. .ai_body_input_textarea::-webkit-scrollbar {
  1334. width: 6px;
  1335. }
  1336. .ai_body_input_textarea::-webkit-scrollbar-track {
  1337. background: #d8d9dc;
  1338. border-radius: 2px;
  1339. }
  1340. .ai_body_input_textarea::-webkit-scrollbar-thumb {
  1341. background: #c9c9c9;
  1342. border-radius: 10px;
  1343. }
  1344. .ai_body_input_textarea::-webkit-scrollbar-thumb:hover {
  1345. background: #c9c9c9;
  1346. }
  1347. .c_pub_button_confirm {
  1348. /* position: absolute;
  1349. bottom: 13px;
  1350. right: 13px; */
  1351. /* margin-top: 10px; */
  1352. width: 60px;
  1353. margin-right: 5px;
  1354. display: flex;
  1355. justify-content: center;
  1356. margin-bottom: 10px;
  1357. position: absolute;
  1358. right: 10px;
  1359. bottom: 0px;
  1360. white-space: nowrap;
  1361. border-radius: 10px;
  1362. }
  1363. .c_pub_button_confirmDisabled {
  1364. background-color: #aeccfe;
  1365. }
  1366. .ai_body_dialog {
  1367. padding: 10px 0;
  1368. box-sizing: border-box;
  1369. /* min-height: calc(20vh - 10px); */
  1370. /* height: calc(100%); */
  1371. overflow: auto;
  1372. margin-bottom: 10px;
  1373. }
  1374. .dialog_content {
  1375. width: 100%;
  1376. display: flex;
  1377. flex-direction: column;
  1378. }
  1379. .dialog_content>div {
  1380. display: flex;
  1381. align-items: flex-start;
  1382. width: 100%;
  1383. }
  1384. .dialog_content+.dialog_content {
  1385. margin: 15px 0;
  1386. }
  1387. .dialog_content>div .right {
  1388. flex-direction: row-reverse;
  1389. }
  1390. .dialog_content>div .right .role {
  1391. margin-right: 0;
  1392. margin-left: 10px;
  1393. }
  1394. .dialog_content>div .role {
  1395. min-width: 30px;
  1396. width: 30px;
  1397. height: 30px;
  1398. margin-right: 10px;
  1399. border-radius: 50%;
  1400. }
  1401. .dialog_content>div .role>img {
  1402. height: 100%;
  1403. width: 100%;
  1404. border-radius: 100%;
  1405. }
  1406. .dialog_content>div .content {
  1407. padding: 10px 10px;
  1408. border-radius: 2px 8px 8px 8px;
  1409. width: auto;
  1410. word-break: break-word;
  1411. box-sizing: border-box;
  1412. /* white-space: pre-line; */
  1413. max-width: calc(100% - 85px);
  1414. background: #f6f9ff;
  1415. /* overflow: hidden; */
  1416. margin: 0 10px;
  1417. position: relative;
  1418. }
  1419. .createTime {
  1420. width: 100%;
  1421. height: 20px;
  1422. position: absolute;
  1423. bottom: -25px;
  1424. left: 0;
  1425. font-size: 14px;
  1426. white-space: nowrap;
  1427. color: #919191;
  1428. }
  1429. .dialog_content>div .content2 {
  1430. background: #3681fc;
  1431. color: #fff;
  1432. border-radius: 8px 2px 8px 8px;
  1433. margin-left: auto;
  1434. }
  1435. .ai_body_select {
  1436. position: relative;
  1437. }
  1438. .ai_body_select>.check {
  1439. background: #e7e7e7;
  1440. display: flex;
  1441. width: fit-content;
  1442. padding: 0 10px;
  1443. height: 30px;
  1444. border-radius: 21px;
  1445. font-size: 14px;
  1446. align-items: center;
  1447. justify-content: center;
  1448. color: #0061ff;
  1449. font-weight: 700;
  1450. margin: 10px 0;
  1451. cursor: pointer;
  1452. }
  1453. .ai_body_select>.check::before {
  1454. content: "";
  1455. width: 15px;
  1456. height: 15px;
  1457. display: block;
  1458. background-image: url("../../../assets/icon/course/aiPart.png");
  1459. background-size: 100% 100%;
  1460. margin-right: 5px;
  1461. }
  1462. .ai_body_select>.check::after {
  1463. content: "";
  1464. width: 15px;
  1465. height: 15px;
  1466. display: block;
  1467. background-image: url("../../../assets/icon/course/aiPart_arrow.png");
  1468. background-size: 100% 100%;
  1469. margin-right: 5px;
  1470. }
  1471. .ai_body_select>.isCheck {
  1472. background: #0061ff;
  1473. display: flex;
  1474. width: fit-content;
  1475. padding: 0 10px;
  1476. height: 30px;
  1477. border-radius: 21px;
  1478. font-size: 14px;
  1479. align-items: center;
  1480. justify-content: center;
  1481. color: #fff;
  1482. font-weight: 700;
  1483. margin: 10px 0;
  1484. cursor: pointer;
  1485. max-width: 100%;
  1486. box-sizing: border-box;
  1487. }
  1488. .ai_body_select>.isCheck>span {
  1489. width: calc(100% - 40px);
  1490. display: block;
  1491. overflow: hidden;
  1492. white-space: nowrap;
  1493. text-overflow: ellipsis;
  1494. }
  1495. .ai_body_select>.isCheck::before {
  1496. content: "";
  1497. width: 15px;
  1498. height: 15px;
  1499. display: block;
  1500. background-image: url("../../../assets/icon/course/aiPart_active.png");
  1501. background-size: 100% 100%;
  1502. margin-right: 5px;
  1503. }
  1504. .ai_body_select>.isCheck::after {
  1505. content: "";
  1506. width: 15px;
  1507. height: 15px;
  1508. display: block;
  1509. background-image: url("../../../assets/icon/course/aiPart_arrow_active.png");
  1510. background-size: 100% 100%;
  1511. margin-right: 5px;
  1512. }
  1513. .ai_body_select>.checkBox {
  1514. position: absolute;
  1515. bottom: 40px;
  1516. border: 1px solid #e0eafb;
  1517. width: 100%;
  1518. height: 300px;
  1519. background: #fff;
  1520. border-radius: 5px;
  1521. padding: 10px;
  1522. box-sizing: border-box;
  1523. }
  1524. .ai_body_select>.checkBox>.task>.title,
  1525. .ai_body_select>.checkBox>.part>.title {
  1526. font-size: 14px;
  1527. font-weight: 700;
  1528. margin-bottom: 5px;
  1529. }
  1530. .ai_body_select>.checkBox>.task {
  1531. height: calc(100% - 60px);
  1532. }
  1533. .ai_body_select>.checkBox>.part {}
  1534. .ai_body_select>.checkBox>.task>.content {
  1535. height: calc(100% - 40px);
  1536. overflow: auto;
  1537. }
  1538. .ai_body_select>.checkBox>.task>.content>.span+.span {
  1539. margin-top: 5px;
  1540. }
  1541. .ai_body_select>.checkBox>.task>.content>.span {
  1542. display: flex;
  1543. align-items: center;
  1544. font-size: 14px;
  1545. cursor: pointer;
  1546. }
  1547. .ai_body_select>.checkBox>.task>.content>.span>.check {
  1548. width: 13px;
  1549. height: 13px;
  1550. display: flex;
  1551. align-items: center;
  1552. margin-right: 5px;
  1553. }
  1554. .ai_body_select>.checkBox>.task>.content>.span>.check>img {
  1555. width: 100%;
  1556. height: 100%;
  1557. }
  1558. .ai_body_select>.checkBox>.part>.content {
  1559. display: flex;
  1560. align-items: center;
  1561. font-size: 14px;
  1562. justify-content: space-between;
  1563. }
  1564. .ai_body_select>.checkBox>.part>.content>.span {
  1565. padding: 3px 6px;
  1566. border: 1px solid #e0eafb;
  1567. border-radius: 40px;
  1568. cursor: pointer;
  1569. }
  1570. .ai_body_select>.checkBox>.part>.content>.span.active {
  1571. color: #0061ff;
  1572. border-color: #0061ff;
  1573. }
  1574. .ai_b_i_jListPanel {
  1575. width: 400px;
  1576. height: 100%;
  1577. top: 0;
  1578. right: 0;
  1579. position: fixed;
  1580. /* background-color: #000; */
  1581. z-index: 990;
  1582. }
  1583. .ai_b_i_jListBox {
  1584. width: 100%;
  1585. max-height: 300px;
  1586. background-color: #fff;
  1587. position: absolute;
  1588. bottom: calc(100% + 5px);
  1589. box-sizing: border-box;
  1590. padding: 10px;
  1591. overflow: auto;
  1592. border-radius: 8px;
  1593. border: 1px solid #e7e7e7;
  1594. box-shadow: 0 4px 10px 0 rgba(29, 57, 131, 0.08),
  1595. 1px 1px 20px 4px rgba(29, 57, 131, 0.05);
  1596. z-index: 999;
  1597. }
  1598. .jlist_box {
  1599. background: #f1f1f1;
  1600. border-radius: 5px;
  1601. margin-bottom: 10px;
  1602. padding: 8px 18px 8px 8px;
  1603. box-sizing: border-box;
  1604. width: 100%;
  1605. position: relative;
  1606. }
  1607. .jlist_box .cancel {
  1608. position: absolute;
  1609. top: 8px;
  1610. right: 5px;
  1611. cursor: pointer;
  1612. font-size: 14px;
  1613. }
  1614. .jlist_box>span {
  1615. display: -webkit-box;
  1616. -webkit-box-orient: vertical;
  1617. -webkit-line-clamp: 3;
  1618. text-overflow: ellipsis;
  1619. overflow: hidden;
  1620. word-break: break-all;
  1621. }
  1622. .ai_btn_box{
  1623. min-width: fit-content;
  1624. margin-top: auto;
  1625. }
  1626. .ai_btn_box > img{
  1627. cursor: pointer;
  1628. width: 15px;
  1629. }
  1630. </style>