aiCreateDialog.vue 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670
  1. <template>
  2. <el-dialog title="AI生成PPT" :visible.sync="dialogVisibleAiCreate" :append-to-body="true" width="700px"
  3. :before-close="handleClose" class="dialog_diy">
  4. <div style="height: 500px; padding: 15px" v-loading="loading" element-loading-text="小可正在努力生成中,请稍等...">
  5. <!-- <div class="t_box">
  6. <span>选择:</span>
  7. <el-radio-group v-model="radio" @change="changeRadio">
  8. <el-radio :label="0">PPT</el-radio>
  9. <el-radio :label="1">教案</el-radio>
  10. <el-radio :label="2">视频</el-radio>
  11. </el-radio-group>
  12. </div> -->
  13. <div class="t_box" v-if="steps == 1" style="height: 100%">
  14. <textarea style="height: 100%;width:calc(100% - 260px)" rows="10" class="binfo_input binfo_textarea" cols
  15. placeholder="请生成大纲" v-model="outline"></textarea>
  16. <div class="template_box">
  17. <span class="title">选择模板</span>
  18. <div class="template_list">
  19. <div class="template_item" :class="{ active: index == templateIndex }"
  20. v-for="(item, index) in templateList" :key="index" @click="changeTemplate(index)">
  21. <img :src="item.img" alt="" />
  22. </div>
  23. </div>
  24. </div>
  25. </div>
  26. <div style="height: 100%" v-else>
  27. <wOffice v-if="url" :url="url"></wOffice>
  28. </div>
  29. </div>
  30. <span slot="footer" class="dialog-footer">
  31. <el-button @click="aiGet(2)" type="primary" :disabled="loading">重新生成大纲</el-button>
  32. <el-button @click="aiGet(1)" type="primary" :disabled="loading">{{
  33. url ? "重新生成PPT" : "生成PPT"
  34. }}</el-button>
  35. <el-button @click="steps = 1" type="primary" v-if="steps == 2">上一步</el-button>
  36. <el-button @click="steps = 2" type="primary" v-else-if="steps == 1 && url">下一步</el-button>
  37. <el-button @click="confirm" type="primary">确 定</el-button>
  38. <el-button @click="close">关 闭</el-button>
  39. </span>
  40. </el-dialog>
  41. </template>
  42. <script>
  43. import Pptxgen from "pptxgenjs";
  44. import wOffice from "../components/wOffice.vue";
  45. import { v4 as uuidv4 } from "uuid";
  46. import { color } from "highcharts";
  47. export default {
  48. components: {
  49. wOffice,
  50. },
  51. props: {
  52. dialogVisibleAiCreate: {
  53. type: Boolean,
  54. default: false,
  55. },
  56. courseName: {
  57. type: String,
  58. default: "",
  59. },
  60. infoData: {
  61. type: Array,
  62. default: "",
  63. },
  64. courseTypeId: {
  65. type: Array,
  66. default: "",
  67. },
  68. CourseTypeJson: {
  69. type: Object,
  70. default: () => ({}),
  71. }
  72. },
  73. // 根据用户给你的参考资料
  74. data() {
  75. return {
  76. userid: this.$route.query.userid,
  77. radio: 0,
  78. aiJson: {
  79. ppt: `## 任务
  80. 请生成关于${this.courseName},为教师生成这节课的教学ppt,页数在20页左右。PPT的内容主要是讲解该课程中所有可能涉及到的知识点。
  81. ## 工作流
  82. 1. 从用户提供的参考资料中提取10个最重要的知识点(知识点水平限制在小学和初中),并输出。
  83. 2. 针对10个知识点中的每个,你使用1~3页ppt详细的对知识点进行讲解。你的讲解词应该在100token左右
  84. 3. 讲解完所有知识点后,再根据知识点出5道单选题(放在5页ppt中)
  85. ## 限制
  86. - 你不能输出错误的知识,如果你实在不清楚,输出“对不起,我不确定”
  87. - 你不能输出违反伦理的内容`,
  88. word: "",
  89. video: "",
  90. },
  91. aiUrl: {
  92. ppt: "",
  93. word: "",
  94. video: "",
  95. },
  96. detail: "",
  97. loading: false,
  98. url: "",
  99. uJson: {},
  100. outline: "",
  101. steps: 1,
  102. templateList: [
  103. // { img: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model1-11719468995661.png', img2: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model1-21719469026755.png',color:'17094F' },
  104. // { img: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model2-11719469051869.png', img2: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model2-21719469040181.png',color:'052B37' },
  105. // { img: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model3-11719469071576.png', img2: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model3-21719469092087.png',color:'1D5869' },
  106. // { img: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model4-11719469106190.png', img2: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model-4-21719469125318.png',color:'372213' },
  107. // { img: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model5-11719295908696.png', img2: 'https://ccrb.s3.cn-northwest-1.amazonaws.com.cn/model5-21719295930345.png',color:'674D40' },
  108. { img: require('../../../assets/icon/ppt/model1-1.png'), img2: require('../../../assets/icon/ppt/model1-2.png'),color:'17094F' },
  109. { img: require('../../../assets/icon/ppt/model2-1.png'), img2: require('../../../assets/icon/ppt/model2-2.png'),color:'052B37' },
  110. { img: require('../../../assets/icon/ppt/model3-1.png'), img2: require('../../../assets/icon/ppt/model3-2.png'),color:'1D5869' },
  111. { img: require('../../../assets/icon/ppt/model4-1.png'), img2: require('../../../assets/icon/ppt/model4-2.png'),color:'372213' },
  112. { img: require('../../../assets/icon/ppt/model5-1.png'), img2: require('../../../assets/icon/ppt/model5-2.png'),color:'674D40' },
  113. ],
  114. templateIndex: 0,
  115. };
  116. },
  117. watch: {
  118. dialogVisibleAiCreate(newValue, oldValue) {
  119. if (newValue) {
  120. this.loading = false;
  121. this.aiGet(2);
  122. }
  123. },
  124. },
  125. methods: {
  126. handleClose(done) {
  127. this.close();
  128. done();
  129. },
  130. close() {
  131. this.$emit("update:dialogVisibleAiCreate", false);
  132. },
  133. confirm() {
  134. if (this.url) {
  135. this.$emit("createAiPpt", this.uJson);
  136. } else {
  137. this.$message.error("请先生成ppt");
  138. }
  139. },
  140. changeRadio() {
  141. if (this.radio == 0) {
  142. this.detail = this.aiJson.ppt;
  143. }
  144. if (this.radio == 1) {
  145. this.detail = this.aiJson.word;
  146. }
  147. if (this.radio == 2) {
  148. this.detail = this.aiJson.video;
  149. }
  150. },
  151. createFileid(url) {
  152. let _this = this;
  153. return new Promise((resolve, reject) => {
  154. try {
  155. _this.ajax
  156. .put("https://gpt4.cocorobo.cn/upload_file_knowledge", {
  157. url: url,
  158. })
  159. .then((res) => {
  160. let _data = res.data.FunctionResponse;
  161. if (_data.result && _data.result.id) {
  162. resolve(_data.result.id);
  163. }
  164. })
  165. .catch(function (error) {
  166. resolve("");
  167. });
  168. } catch (e) {
  169. resolve();
  170. }
  171. });
  172. },
  173. async aiGet(type) {
  174. if (this.loading) {
  175. this.$message.error("正在生成中,请稍后");
  176. return;
  177. }
  178. this.url = "";
  179. this.uJson = {};
  180. let _this = this;
  181. let fileid = [];
  182. if (_this.infoData.length) {
  183. for (var i = 0; i < _this.infoData.length; i++) {
  184. if (_this.infoData[i].fileid) {
  185. fileid.push(_this.infoData[i].fileid);
  186. } else {
  187. let _fileid = await _this.createFileid(_this.infoData[i].url);
  188. if (_fileid) {
  189. _this.infoData[i].fileid = _fileid;
  190. _this.$forceUpdate();
  191. fileid.push(_fileid);
  192. }
  193. }
  194. }
  195. }
  196. console.log("fileid=========", fileid);
  197. let mclass = [];
  198. if (_this.courseTypeId.length) {
  199. for (var i = 0; i < _this.courseTypeId.length; i++) {
  200. let _sid = _this.courseTypeId[i];
  201. for (
  202. var j = 0;
  203. j <
  204. _this.CourseTypeJson["34628934-d02f-11ec-8c78-005056b86db5"].length;
  205. j++
  206. ) {
  207. if (
  208. _sid ==
  209. _this.CourseTypeJson["34628934-d02f-11ec-8c78-005056b86db5"][j].id
  210. ) {
  211. mclass.push(
  212. _this.CourseTypeJson["34628934-d02f-11ec-8c78-005056b86db5"][j]
  213. .name
  214. );
  215. }
  216. }
  217. }
  218. }
  219. let message = "";
  220. if (type == 1) {
  221. message = `NOTICE
  222. Role: 你是ppt内容设计大师,能力是从用户提供的文件资料中提取最重要的学科概念作为ppt参考内容,最后根据Context要求的流程要求输出ppt内容。
  223. Output: Provide your output in json format.
  224. Language: Please use the same language as the user requirement, if the user speaks Chinese, the specific text of your answer should also be in Chinese.
  225. ATTENTION: Use '##' to SPLIT SECTIONS, not '#'. Output format carefully referenced "Format example".
  226. Instruction: Based on the context, follow "Format example", write content.
  227. # Context
  228. ## 任务
  229. 将参考#大纲内容,为教师生成这节课的教学ppt。PPT的内容主要是讲解该课程中所有可能涉及到的知识点,根据大纲内容生成ppt内容。
  230. ## 每一页输出格式
  231. - 页数:序列数字
  232. - 标题:学科概念(请从给你的大纲中摘取)
  233. - 子标题:知识点(请从给你的大纲中摘取)
  234. - 知识点讲解:针对大纲中的每个知识点,生成200字左右的详细讲解。你的语气应该让小学或初中的学生清晰易懂的讲解。你的讲解词在200 token左右。请尽可能的详细,这对我很重要。
  235. ## 限制
  236. - 你不能输出错误的知识,如果你实在不清楚,修改大纲中的知识点。
  237. - 你不能输出违反伦理的内容
  238. ## 工作流
  239. 1. 针对大纲中的每个知识点,生成200字左右的详细讲解。你的语气应该让小学或初中的学生清晰易懂的讲解。请尽可能的详细,这对我很重要。
  240. 2. 针对大纲中的每个测试,详细设计不同测试题目,例如单选,多选,对错题等。
  241. 3.从用户提供的参考资料中提取5个最重要的学科概念,并输出。
  242. 4.分解每个学科概念为几个子知识点
  243. 5.简要描述每个知识点
  244. 6.生成5个测试题以考察学生的掌握情况
  245. 7.一个知识点一页,一个测试题一页
  246. ## 大纲内容
  247. ${_this.outline.replaceAll('#','').replaceAll('*','').replaceAll('-','').replaceAll('\n','')}
  248. # Format example
  249. [{"page": "页码(数字)","title": "学科概念(请从给你的大纲中摘取)(标题)","task": "知识点(请从给你的大纲中摘取)(子标题)","points": "知识点讲解:针对大纲中的每个知识点,生成200字左右的详细讲解。你的语气应该让小学或初中的学生清晰易懂的讲解。你的讲解词在100 token左右。请尽可能的详细,这对我很重要。"}]`;
  250. } else {
  251. message = `# 任务
  252. 请根据参考资料,生成关于${this.courseName},为教师生成这节课的教学ppt的大纲,大纲的主要内容课程知识点的讲解与相关练习和测试。你的输出应该符合#输出格式
  253. # 工作流
  254. 1.从用户提供的参考资料中提取5个最重要的学科概念${mclass.length ? "(水平限制在{面向年级}中)" : ""},并输出。
  255. 2.分解每个学科概念为几个子知识点
  256. 3.简要描述每个知识点
  257. 4.生成5个测试题以考察学生的掌握情况
  258. ${mclass.length ? "#参考资料\n面向年级:" + mclass.join(",") : ""}
  259. # 输出格式
  260. - 标题:学科概念1
  261. 1.知识点:知识点1
  262. 2.知识点:知识点2
  263. 3.知识点:知识点3
  264. 4.知识点:知识点4
  265. 5.知识点:知识点5
  266. # 限制
  267. 1.如果有参考资料请根据参考资料,如果没有无需根据参考资料进行,随意发挥。
  268. 2.你不能输出错误的知识。
  269. 3.你不能输出违反伦理的内容。
  270. `;
  271. }
  272. // let params = JSON.stringify({
  273. // // "model": "Chat",
  274. // model: 'gpt-3.5-turbo',
  275. // temperature: 0,
  276. // max_tokens: 4096,
  277. // top_p: 1,
  278. // frequency_penalty: 0,
  279. // presence_penalty: 0,
  280. // messages: [{
  281. // content: message,
  282. // role: 'user'
  283. // }],
  284. // stream: false,
  285. // uid: this.userid,
  286. // mind_map_question: "",
  287. // })
  288. // _this.loading = true
  289. // _this.ajax.post('https://gpt4.cocorobo.cn/chat', params).then(function (response) {
  290. // console.log(response);
  291. // let data = response.data.FunctionResponse
  292. // if (data.choices && data.choices.length && data.choices[0].message) {
  293. // console.log(data.choices[0].message.content);
  294. // try {
  295. // let _data = JSON.parse(data.choices[0].message.content)
  296. // _this.createPpt(_data)
  297. // } catch (e) {
  298. // console.log('error_________________'+e);
  299. // _this.$message.error(data.choices[0].message.content)
  300. // _this.loading = false
  301. // }
  302. // }
  303. // }).catch(function (error) {
  304. // _this.loading = false
  305. // console.log(error);
  306. // });
  307. let parm = {
  308. assistant_id:
  309. type == 1
  310. ? "6063369f-289a-11ef-8bf4-12e77c4cb76b"
  311. : "f8e1ebb2-2e0d-11ef-8bf4-12e77c4cb76b",
  312. message: [{ type: "text", text: message }],
  313. session_name: uuidv4(),
  314. userId: this.userid,
  315. file_ids: fileid.length ? [...fileid] : "",
  316. };
  317. _this.loading = true;
  318. this.ajax
  319. .post("https://gpt4.cocorobo.cn/ai_agent_park_chat", parm)
  320. .then((response) => {
  321. console.log(response);
  322. let data = response.data.FunctionResponse;
  323. if (data.message) {
  324. console.log(data.message);
  325. if (type == 1) {
  326. try {
  327. let _data = JSON.parse(
  328. data.message.replaceAll("```json", "").replaceAll("```", "")
  329. );
  330. _this.createPpt(_data);
  331. _this.steps = 2;
  332. } catch (e) {
  333. console.log("error_________________" + e);
  334. // _this.$message.error(data.message)
  335. try {
  336. let regex = new RegExp("(?<=```json)([\\s\\S]*?)(?=```)");
  337. let match = data.message.match(regex);
  338. _data2 = JSON.parse(match[0]);
  339. _this.createPpt(_data2);
  340. _this.steps = 2;
  341. // let _data = JSON.parse(data.message.match(/(?<=```json).*?(?=```)/)[0])
  342. // var message = data.message;
  343. // var jsonStart = message.indexOf("```json") + 7; // `+ 7` 是为了跳过 ```json
  344. // var jsonEnd = message.indexOf("```", jsonStart);
  345. // if (jsonStart !== -1 && jsonEnd !== -1) {
  346. // var jsonString = message
  347. // .substring(jsonStart, jsonEnd)
  348. // .trim();
  349. // var _data2 = JSON.parse(jsonString);
  350. // _this.createPpt(_data2);
  351. // _this.steps = 2;
  352. // } else {
  353. // _this.$message.error("生成失败,正在重新生成");
  354. // _this.aiGet(type);
  355. // }
  356. } catch (error) {
  357. _this.$message.error("生成失败,正在重新生成");
  358. _this.loading = false;
  359. _this.aiGet(type);
  360. }
  361. }
  362. } else {
  363. _this.outline = data.message;
  364. _this.steps = 1;
  365. _this.loading = false;
  366. }
  367. }
  368. })
  369. .catch((error) => {
  370. _this.loading = false;
  371. console.log(error);
  372. });
  373. },
  374. createPpt(array) {
  375. // 1. 创建PPT
  376. const pres = new Pptxgen();
  377. const _slideTou = pres.addSlide();
  378. _slideTou.background = { path: this.templateList[this.templateIndex].img }
  379. _slideTou.addText(this.courseName, {
  380. x: "10%", // 横坐标
  381. y: 3,
  382. width: "90%",
  383. color: this.templateList[this.templateIndex].color,
  384. fontSize: 38, // 字号
  385. align: "center",
  386. });
  387. for (var i = 0; i < array.length; i++) {
  388. // 2. 创建一个PPT页面,每调用一次 pres.addSlide() 都可以生成一张新的页面
  389. // 建议把每个页面的构造抽成一个个函数,然后通过函数调用生成新页面,代码不会很乱
  390. const _slide = pres.addSlide();
  391. _slide.background = { path: this.templateList[this.templateIndex].img2 }
  392. // 3. 调用addTetx(),在PPT页面中插入文字“Hello World from PptxGenJS...”
  393. // 括号里面是对文字的配置,文字横坐标x为1.5,纵坐标y为1.5,字体颜色 363636……
  394. // 关于坐标长度与px的转换 x 1 = 127~128px 左右
  395. const page = i + 1 > 10 ? i + 1 : "0" + (i + 1);
  396. const tempResult1 = page + " " + array[i].title;
  397. _slide.addText(tempResult1, {
  398. x: "10%", // 横坐标
  399. y: 1.1,
  400. width: "90%",
  401. color: this.templateList[this.templateIndex].color,
  402. fontSize: 28, // 字号
  403. align: "center",
  404. });
  405. const tempResult2 = array[i].task;
  406. _slide.addText(tempResult2, {
  407. x: "10%", // 横坐标
  408. y: 1.8,
  409. width: "90%",
  410. color: "222222",
  411. fontSize: 20, // 字号
  412. align: "center",
  413. });
  414. let tempResult3 = '';
  415. if (typeof array[i].points == 'object') {
  416. tempResult3 = array[i].points.join('\n')
  417. } else {
  418. tempResult3 = array[i].points
  419. }
  420. _slide.addText(tempResult3, {
  421. x: "20%", // 横坐标
  422. y: 3.2,
  423. w: "60%",
  424. color: "444444",
  425. fontSize: 18, // 字号
  426. align: "center",
  427. });
  428. }
  429. const _slideWei = pres.addSlide();
  430. _slideWei.background = { path: this.templateList[this.templateIndex].img }
  431. _slideWei.addText("谢谢观看,下课!", {
  432. x: "10%", // 横坐标
  433. y: 3,
  434. width: "90%",
  435. color: this.templateList[this.templateIndex].color,
  436. fontSize: 38, // 字号
  437. align: "center",
  438. });
  439. // 获取PPTX文件的ArrayBuffer
  440. // 保存为 Blob 并处理
  441. pres.write("blob").then((blob) => {
  442. // 现在你有了一个 Blob 对象
  443. console.log(blob);
  444. const file = new File([blob], this.courseName + ".pptx", {
  445. type: "application/vnd.openxmlformats-officedocument.presentationml.presentation",
  446. });
  447. console.log(pres);
  448. this.beforeUpload(file);
  449. });
  450. },
  451. beforeUpload(event) {
  452. var file = event;
  453. var credentials = {
  454. accessKeyId: "AKIATLPEDU37QV5CHLMH",
  455. secretAccessKey: "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",
  456. }; //秘钥形式的登录上传
  457. window.AWS.config.update(credentials);
  458. window.AWS.config.region = "cn-northwest-1"; //设置区域
  459. var bucket = new window.AWS.S3({ params: { Bucket: "ccrb" } }); //选择桶
  460. var _this = this;
  461. if (file) {
  462. var params = {
  463. Key:
  464. file.name.split(".")[0] +
  465. new Date().getTime() +
  466. "." +
  467. file.name.split(".")[file.name.split(".").length - 1],
  468. ContentType: file.type,
  469. Body: file,
  470. "Access-Control-Allow-Credentials": "*",
  471. ACL: "public-read",
  472. }; //key可以设置为桶的相抵路径,Body为文件, ACL最好要设置
  473. var options = {
  474. partSize: 2048 * 1024 * 1024,
  475. queueSize: 2,
  476. leavePartsOnError: true,
  477. };
  478. bucket
  479. .upload(params, options)
  480. .on("httpUploadProgress", function (evt) {
  481. //这里可以写进度条
  482. // console.log("Uploaded : " + parseInt((evt.loaded * 80) / evt.total) + '%');
  483. })
  484. .send(function (err, data) {
  485. _this.loading = false;
  486. if (err) {
  487. _this.$message.error("上传失败");
  488. } else {
  489. _this.url = data.Location;
  490. _this.uJson = {
  491. name: file.name,
  492. url: data.Location,
  493. type: 3,
  494. };
  495. console.log(data.Location);
  496. }
  497. });
  498. }
  499. },
  500. changeTemplate(index) {
  501. this.templateIndex = index;
  502. }
  503. },
  504. };
  505. </script>
  506. <style scoped>
  507. .dialog_diy>>>.el-dialog {
  508. height: auto;
  509. margin: 15vh auto 0 !important;
  510. }
  511. .dialog_diy>>>.el-dialog__header {
  512. background: #454545 !important;
  513. padding: 15px 20px;
  514. }
  515. .dialog_diy>>>.el-dialog__body {
  516. height: calc(100% - 124px);
  517. box-sizing: border-box;
  518. padding: 0px;
  519. }
  520. .dialog_diy>>>.el-dialog__title {
  521. color: #fff;
  522. }
  523. .dialog_diy>>>.el-dialog__headerbtn {
  524. top: 19px;
  525. }
  526. .dialog_diy>>>.el-dialog__headerbtn .el-dialog__close {
  527. color: #fff;
  528. }
  529. .dialog_diy>>>.el-dialog__headerbtn .el-dialog__close:hover {
  530. color: #fff;
  531. }
  532. .dialog_diy>>>.el-dialog__body,
  533. .dialog_diy>>>.el-dialog__footer {
  534. background: #fafafa;
  535. }
  536. .binfo_input {
  537. width: 100%;
  538. margin: 0;
  539. padding: 5px 7px;
  540. display: block;
  541. min-width: 0;
  542. outline: none;
  543. box-sizing: border-box;
  544. background: none;
  545. border: none;
  546. border-radius: 4px;
  547. background: #fff;
  548. font-size: 15px;
  549. resize: none;
  550. font-family: "Microsoft YaHei";
  551. min-height: 48px;
  552. /* border: 1px solid #3682fc00; */
  553. border: 1.5px solid #cad1dc;
  554. }
  555. .binfo_textarea {
  556. border: 1.5px solid #cad1dc;
  557. font-size: 15px;
  558. resize: none;
  559. /* background: #f6f6f6; */
  560. font-family: "Microsoft YaHei";
  561. }
  562. .binfo_textarea:focus-visible {
  563. border: 1.5px solid #3681fc !important;
  564. }
  565. .binfo_textarea::-webkit-scrollbar {
  566. /*滚动条整体样式*/
  567. width: 6px;
  568. /*高宽分别对应横竖滚动条的尺寸*/
  569. height: 6px;
  570. }
  571. /*定义滚动条轨道 内阴影+圆角*/
  572. .binfo_textarea::-webkit-scrollbar {
  573. border-radius: 10px;
  574. background-color: #eee;
  575. }
  576. /*定义滑块 内阴影+圆角*/
  577. .binfo_textarea::-webkit-scrollbar-thumb {
  578. border-radius: 10px;
  579. -webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.3);
  580. background-color: rgba(0, 0, 0, 0.1);
  581. }
  582. .t_box {
  583. display: flex;
  584. margin-bottom: 15px;
  585. display: flex;
  586. justify-content: space-between;
  587. }
  588. .t_box>span:nth-child(1) {
  589. min-width: 80px;
  590. font-size: 16px;
  591. color: #000;
  592. }
  593. .template_box{
  594. width: 250px;
  595. }
  596. .template_box > .title{
  597. font-size: 18px;
  598. margin-bottom: 10px;
  599. display: block;
  600. }
  601. .template_list{
  602. overflow: auto;
  603. width: 100%;
  604. height: calc(100% - 31px);
  605. }
  606. .template_item{
  607. cursor: pointer;
  608. width: 100%;
  609. height: 165px;
  610. overflow: hidden;
  611. border-radius: 5px;
  612. box-sizing: border-box;
  613. border: 5px solid #e5e5e5ee;
  614. }
  615. .template_item + .template_item{
  616. margin-top: 10px;
  617. }
  618. .template_item.active{
  619. border: 5px solid #0061ff;
  620. }
  621. .template_item > img{
  622. width: 100%;
  623. height: 100%;
  624. object-fit: cover;
  625. }
  626. </style>