qqsl-agent 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (70) hide show
  1. package/README.md +24 -0
  2. package/assets/images/card/copy.png +0 -0
  3. package/assets/images/card/refresh.png +0 -0
  4. package/assets/images/card/share.png +0 -0
  5. package/assets/images/card/think-finish.png +0 -0
  6. package/assets/images/input/relate.png +0 -0
  7. package/assets/images/input/send-gray.png +0 -0
  8. package/assets/images/input/send.png +0 -0
  9. package/assets/images/input/toggle-model.png +0 -0
  10. package/assets/images/logo.png +0 -0
  11. package/assets/images/sidebar/collapse.png +0 -0
  12. package/assets/images/sidebar/delete.png +0 -0
  13. package/assets/images/sidebar/expand.png +0 -0
  14. package/assets/images/sidebar/history-chat.png +0 -0
  15. package/assets/images/sidebar/new-chat.png +0 -0
  16. package/assets/images/sidebar/time.png +0 -0
  17. package/assets/styles/index.css +1 -0
  18. package/chat/chat-input/chat-input.component.d.ts +46 -0
  19. package/chat/chat-messages/chat-messages.component.d.ts +33 -0
  20. package/chat/chat-sidebar/chat-sidebar.component.d.ts +20 -0
  21. package/chat/chat-sidebar/history-group/history-group.component.d.ts +18 -0
  22. package/chat/chat.component.d.ts +34 -0
  23. package/chat/chat.module.d.ts +28 -0
  24. package/chat/components/pagination/pagination.component.d.ts +14 -0
  25. package/chat/index.d.ts +5 -0
  26. package/chat/models/conversation.model.d.ts +20 -0
  27. package/chat/models/knowledge-list.model.d.ts +8 -0
  28. package/chat/models/message.model.d.ts +41 -0
  29. package/chat/models/model-list.model.d.ts +9 -0
  30. package/chat/models/send-message.model.d.ts +33 -0
  31. package/chat/pipes/markdown.pipe.d.ts +8 -0
  32. package/chat/public-api.d.ts +2 -0
  33. package/chat/services/http.service.d.ts +13 -0
  34. package/chat/services/http2.service.d.ts +45 -0
  35. package/chat/services/markdown-stream.service.d.ts +10 -0
  36. package/chat/services/sse.service.d.ts +28 -0
  37. package/chat/store/store.service.d.ts +17 -0
  38. package/esm2020/chat/chat-input/chat-input.component.mjs +151 -0
  39. package/esm2020/chat/chat-messages/chat-messages.component.mjs +110 -0
  40. package/esm2020/chat/chat-sidebar/chat-sidebar.component.mjs +118 -0
  41. package/esm2020/chat/chat-sidebar/history-group/history-group.component.mjs +124 -0
  42. package/esm2020/chat/chat.component.mjs +138 -0
  43. package/esm2020/chat/chat.module.mjs +98 -0
  44. package/esm2020/chat/components/pagination/pagination.component.mjs +91 -0
  45. package/esm2020/chat/models/conversation.model.mjs +2 -0
  46. package/esm2020/chat/models/knowledge-list.model.mjs +2 -0
  47. package/esm2020/chat/models/message.model.mjs +35 -0
  48. package/esm2020/chat/models/model-list.model.mjs +2 -0
  49. package/esm2020/chat/models/send-message.model.mjs +2 -0
  50. package/esm2020/chat/pipes/markdown.pipe.mjs +24 -0
  51. package/esm2020/chat/public-api.mjs +3 -0
  52. package/esm2020/chat/qqsl-agent-chat.mjs +5 -0
  53. package/esm2020/chat/services/http.service.mjs +68 -0
  54. package/esm2020/chat/services/http2.service.mjs +101 -0
  55. package/esm2020/chat/services/markdown-stream.service.mjs +137 -0
  56. package/esm2020/chat/services/sse.service.mjs +107 -0
  57. package/esm2020/chat/store/store.service.mjs +38 -0
  58. package/esm2020/public-api.mjs +5 -0
  59. package/esm2020/qqsl-agent.mjs +5 -0
  60. package/fesm2015/qqsl-agent-chat.mjs +1255 -0
  61. package/fesm2015/qqsl-agent-chat.mjs.map +1 -0
  62. package/fesm2015/qqsl-agent.mjs +9 -0
  63. package/fesm2015/qqsl-agent.mjs.map +1 -0
  64. package/fesm2020/qqsl-agent-chat.mjs +1248 -0
  65. package/fesm2020/qqsl-agent-chat.mjs.map +1 -0
  66. package/fesm2020/qqsl-agent.mjs +9 -0
  67. package/fesm2020/qqsl-agent.mjs.map +1 -0
  68. package/index.d.ts +5 -0
  69. package/package.json +45 -0
  70. package/public-api.d.ts +2 -0
@@ -0,0 +1,151 @@
1
+ // chat-input.component.ts
2
+ import { Component, Output, EventEmitter, ViewChild, inject, Input } from '@angular/core';
3
+ import { HttpService } from '../services/http.service';
4
+ import { StoreService } from '../store/store.service';
5
+ import { SseService } from '../services/sse.service';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common";
8
+ import * as i2 from "ng-zorro-antd/core/transition-patch";
9
+ import * as i3 from "ng-zorro-antd/popover";
10
+ import * as i4 from "ng-zorro-antd/radio";
11
+ import * as i5 from "ng-zorro-antd/checkbox";
12
+ import * as i6 from "ng-zorro-antd/icon";
13
+ import * as i7 from "@angular/forms";
14
+ export class ChatInputComponent {
15
+ constructor() {
16
+ this.messages = [];
17
+ this.sendMessage = new EventEmitter();
18
+ this.messageChange = new EventEmitter();
19
+ this.enter = new EventEmitter();
20
+ this.paste = new EventEmitter();
21
+ this.httpService = inject(HttpService);
22
+ this.storeService = inject(StoreService);
23
+ this.sseService = inject(SseService);
24
+ this.isEmpty = true;
25
+ this.currentModel = null;
26
+ this.currentKnowledge = [];
27
+ this.modelList = [];
28
+ this.knowledgeList = [];
29
+ this.modelPopoverVisible = false;
30
+ this.knowledgePopoverVisible = false;
31
+ this.deepThink = true;
32
+ }
33
+ ngOnInit() {
34
+ this.getModelList();
35
+ this.getKnowledgeList();
36
+ }
37
+ // 可供交互的模型
38
+ getModelList() {
39
+ this.httpService.get(`${this.storeService.getBaseUrl()}/api/ChatModel/models`).subscribe((res) => {
40
+ this.modelList = res;
41
+ if (res.length) {
42
+ this.currentModel = res[0];
43
+ }
44
+ });
45
+ }
46
+ // 模型可选的知识库
47
+ getKnowledgeList() {
48
+ this.httpService
49
+ .get(`${this.storeService.getBaseUrl()}/api/ChatModel/knowledges`)
50
+ .subscribe((res) => {
51
+ this.knowledgeList = res;
52
+ });
53
+ }
54
+ modelVisibleChange(value) {
55
+ this.modelPopoverVisible = value;
56
+ }
57
+ knowledgeVisibleChange(value) {
58
+ this.knowledgePopoverVisible = value;
59
+ }
60
+ knowledgeSelectChange(value) {
61
+ this.currentKnowledge = value;
62
+ }
63
+ toggleDeepThink() {
64
+ this.deepThink = !this.deepThink;
65
+ }
66
+ updateEmptyState() {
67
+ const el = this.editor.nativeElement;
68
+ const text = el.textContent?.trim() || '';
69
+ const hasOnlyBr = el.innerHTML === '<br>' || el.innerHTML === '';
70
+ this.isEmpty = text === '' && hasOnlyBr;
71
+ }
72
+ handleInput() {
73
+ this.updateEmptyState();
74
+ const html = this.editor.nativeElement.innerHTML;
75
+ const text = this.editor.nativeElement.textContent || '';
76
+ // this.messageChange.emit(text);
77
+ }
78
+ handleEnter(event) {
79
+ if (!event.shiftKey) {
80
+ event.preventDefault();
81
+ // this.enter.emit((event as any).target!.innerText!.replace(/\s+/g, " ").trim());
82
+ this.send();
83
+ }
84
+ }
85
+ handlePaste(event) {
86
+ // this.paste.emit(event);
87
+ setTimeout(() => this.updateEmptyState(), 0);
88
+ }
89
+ clearInput() {
90
+ this.editor.nativeElement.innerHTML = '';
91
+ this.handleInput();
92
+ }
93
+ regenerateSend(msg) {
94
+ // 正在回答中
95
+ if (this.sseService.loadingSource.value) {
96
+ return;
97
+ }
98
+ const conversation = this.storeService.getConversationICode();
99
+ const message = {
100
+ conversation,
101
+ questionICode: msg.iCode,
102
+ model: this.currentModel?.iCode ?? '',
103
+ think: this.deepThink,
104
+ question: msg.question,
105
+ knowledge: !!this.currentKnowledge.length,
106
+ knowledgeList: this.currentKnowledge,
107
+ };
108
+ this.sendMessage.emit({ message, isRegenerate: true });
109
+ }
110
+ send() {
111
+ // 正在回答中
112
+ if (this.sseService.loadingSource.value) {
113
+ return;
114
+ }
115
+ const text = this.editor.nativeElement.textContent || '';
116
+ const conversation = this.storeService.getConversationICode();
117
+ const message = {
118
+ conversation,
119
+ model: this.currentModel?.iCode ?? '',
120
+ think: this.deepThink,
121
+ question: text,
122
+ knowledge: !!this.currentKnowledge.length,
123
+ knowledgeList: this.currentKnowledge,
124
+ };
125
+ this.clearInput();
126
+ this.sendMessage.emit({ message, isRegenerate: false });
127
+ }
128
+ }
129
+ ChatInputComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ChatInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
130
+ ChatInputComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: ChatInputComponent, selector: "ngx-chat-input", inputs: { messages: "messages" }, outputs: { sendMessage: "sendMessage", messageChange: "messageChange", enter: "enter", paste: "paste" }, viewQueries: [{ propertyName: "editor", first: true, predicate: ["editorRef"], descendants: true }, { propertyName: "chatInputWrapper", first: true, predicate: ["chatInputWrapper"], descendants: true }], ngImport: i0, template: "<!-- \u65B0\u804A\u5929\u4F4D\u4E8E\u5C4F\u5E55\u4E2D\u95F4\uFF0C\u53D1\u9001\u6D88\u606F\u4E4B\u540E\u4F4D\u4E8E\u5E95\u90E8 -->\n<div class=\"chat-input-wrapper\" #chatInputWrapper [class.center]=\"!messages.length\">\n <div class=\"new-chat-title\" *ngIf=\"!messages.length\">\n <img src=\"/assets/images/logo.png\" />\n <span>\u4F60\u597D\uFF0C\u6709\u4EC0\u4E48\u53EF\u4EE5\u5E2E\u5230\u4F60\uFF1F\u5F00\u59CB\u5BF9\u8BDD\u5427</span>\n </div>\n <div class=\"chat-input\">\n <div\n #editorRef\n class=\"input-creative-editor\"\n contenteditable=\"true\"\n spellcheck=\"false\"\n data-placeholder=\"\u8BF7\u8F93\u5165\u5E76\u53D1\u9001\u6D88\u606F\"\n [attr.data-empty]=\"isEmpty\"\n (input)=\"handleInput()\"\n (keydown.enter)=\"handleEnter($event)\"\n (paste)=\"handlePaste($event)\"\n ></div>\n <div class=\"input-bottom\">\n <div class=\"input-tools\">\n <div class=\"select-tool\" [ngClass]=\"{ 'select-tool-active': deepThink }\" (click)=\"toggleDeepThink()\">\n <svg\n t=\"1764131026949\"\n style=\"width: 14px; height: 14px\"\n class=\"icon\"\n viewBox=\"0 0 1024 1024\"\n version=\"1.1\"\n xmlns=\"http://www.w3.org/2000/svg\"\n p-id=\"5012\"\n width=\"64\"\n height=\"64\"\n >\n <path\n d=\"M903.36 512a795.52 795.52 0 0 1 65.536 136.704c48 132.16 39.616 230.528-25.024 295.168-64.64 64.64-163.008 72.96-295.168 24.96A795.52 795.52 0 0 1 512 903.424a795.52 795.52 0 0 1-136.704 65.536c-132.096 48-230.528 39.616-295.168-25.024-64.64-64.64-72.96-163.008-24.96-295.104 16.64-46.016 38.528-91.52 65.536-136.768A795.52 795.52 0 0 1 55.04 375.232C7.04 243.2 15.36 144.768 80 80.128 144.768 15.488 243.2 7.168 375.296 55.04c46.016 16.64 91.584 38.528 136.768 65.536A795.52 795.52 0 0 1 648.768 55.104c132.096-48 230.464-39.616 295.104 25.024 64.64 64.64 72.96 163.008 25.024 295.104A795.52 795.52 0 0 1 903.36 512z m-53.12-79.424c15.168-28.736 27.968-57.536 38.464-86.4 35.584-98.112 33.92-166.656-5.12-205.696-39.04-39.04-107.648-40.768-205.696-5.12a685.44 685.44 0 0 0-86.4 38.4 1240.96 1240.96 0 0 1 138.432 120.32 1240.832 1240.832 0 0 1 120.32 138.496zM432.576 173.824a685.44 685.44 0 0 0-86.4-38.528c-98.112-35.584-166.656-33.92-205.696 5.12-39.04 39.04-40.768 107.648-5.12 205.696 10.432 28.928 23.296 57.728 38.4 86.4a1240.896 1240.896 0 0 1 120.32-138.432 1240.832 1240.832 0 0 1 138.496-120.32zM173.824 591.488a685.44 685.44 0 0 0-38.464 86.4c-35.648 98.048-33.92 166.592 5.12 205.632 39.04 39.04 107.584 40.768 205.696 5.12a685.44 685.44 0 0 0 86.4-38.4 1240.768 1240.768 0 0 1-138.496-120.32 1240.96 1240.96 0 0 1-120.256-138.432z m495.744 78.08A1112.064 1112.064 0 0 0 802.048 512a1112.064 1112.064 0 0 0-132.48-157.568A1112.128 1112.128 0 0 0 512 221.952a1112.128 1112.128 0 0 0-157.504 132.48A1112.064 1112.064 0 0 0 222.016 512a1112.192 1112.192 0 0 0 132.416 157.568A1112.064 1112.064 0 0 0 512 802.048a1112.128 1112.128 0 0 0 157.568-132.48z m-78.08 180.608c28.672 15.168 57.472 28.032 86.4 38.464 98.048 35.648 166.592 33.92 205.632-5.12 39.04-39.04 40.768-107.584 5.12-205.696a685.504 685.504 0 0 0-38.4-86.4 1240.832 1240.832 0 0 1-120.32 138.496 1240.96 1240.96 0 0 1-138.432 120.32zM585.088 512a73.152 73.152 0 1 1-146.24 0 73.152 73.152 0 0 1 146.304 0z\"\n fill=\"currentColor\"\n p-id=\"5013\"\n ></path>\n </svg>\n \u6DF1\u5EA6\u601D\u8003\n </div>\n <div\n class=\"select-tool\"\n nz-popover\n [nzPopoverContent]=\"modelListContent\"\n nzPopoverTrigger=\"click\"\n nzPopoverOverlayClassName=\"params-select-popover\"\n [nzPopoverVisible]=\"modelPopoverVisible\"\n (nzPopoverVisibleChange)=\"modelVisibleChange($event)\"\n >\n <img src=\"/assets/images/input/toggle-model.png\" style=\"width: 12px; height: 12px\" />\n {{ currentModel ? currentModel?.name : '\u9009\u62E9\u6A21\u578B' }}\n </div>\n <div\n class=\"select-tool\"\n nz-popover\n [nzPopoverContent]=\"knowledgeListContent\"\n nzPopoverTrigger=\"click\"\n nzPopoverOverlayClassName=\"params-select-popover\"\n [nzPopoverVisible]=\"knowledgePopoverVisible\"\n (nzPopoverVisibleChange)=\"knowledgeVisibleChange($event)\"\n >\n <img src=\"/assets/images/input/relate.png\" style=\"width: 14px; height: 8px\" />\n \u5173\u8054\u77E5\u8BC6\u5E93\n </div>\n </div>\n <img class=\"send\" src=\"/assets/images/input/send.png\" (click)=\"send()\" [hidden]=\"sseService.loading$ | async\" />\n <div class=\"send-loading\" *ngIf=\"sseService.loading$ | async\">\n <div class=\"send-loading-inner\">\n <svg viewBox=\"0 0 36 36\" version=\"1.1\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\" data-icon=\"spin\">\n <defs>\n <linearGradient x1=\"0%\" y1=\"100%\" x2=\"100%\" y2=\"100%\" id=\"linearGradient-1\">\n <stop stop-color=\"currentColor\" stop-opacity=\"0\" offset=\"0%\"></stop>\n <stop stop-color=\"currentColor\" stop-opacity=\"0.50\" offset=\"39.9430698%\"></stop>\n <stop stop-color=\"currentColor\" offset=\"100%\"></stop>\n </linearGradient>\n </defs>\n <g stroke=\"none\" stroke-width=\"1\" fill=\"none\" fill-rule=\"evenodd\">\n <rect fill-opacity=\"0.01\" fill=\"none\" x=\"0\" y=\"0\" width=\"36\" height=\"36\"></rect>\n <path\n d=\"M34,18 C34,9.163444 26.836556,2 18,2 C11.6597233,2 6.18078805,5.68784135 3.59122325,11.0354951\"\n stroke=\"url(#linearGradient-1)\"\n stroke-width=\"4\"\n stroke-linecap=\"round\"\n ></path>\n </g>\n </svg>\n </div>\n </div>\n <!-- <img class=\"send\" src=\"/assets/images/input/send-gray.png\" /> -->\n </div>\n </div>\n <div class=\"chat-input-backdrop\"></div>\n</div>\n\n<ng-template #modelListContent>\n <div style=\"width: 200px\">\n <div class=\"select-title\">\n \u5207\u6362\u6A21\u578B\n <i nz-icon nzType=\"close\" style=\"color: #999\" (click)=\"modelVisibleChange(false)\"></i>\n </div>\n <div class=\"select-content\">\n <nz-radio-group [(ngModel)]=\"currentModel\">\n <label\n nz-radio\n [nzValue]=\"item\"\n [style.margin-top.px]=\"i === 0 ? 0 : 10\"\n *ngFor=\"let item of modelList; let i = index\"\n >\n {{ item.name }}\n </label>\n </nz-radio-group>\n </div>\n </div>\n</ng-template>\n\n<ng-template #knowledgeListContent>\n <div style=\"width: 150px\">\n <div class=\"select-title\">\n \u5173\u8054\u77E5\u8BC6\u5E93\n <i nz-icon nzType=\"close\" style=\"color: #999\" (click)=\"knowledgeVisibleChange(false)\"></i>\n </div>\n <div class=\"select-content\">\n <nz-checkbox-wrapper style=\"width: 100%\" (nzOnChange)=\"knowledgeSelectChange($event)\">\n <div *ngFor=\"let item of knowledgeList\">\n <label nz-checkbox [nzValue]=\"item.iCode\">{{ item.name }}</label>\n </div>\n </nz-checkbox-wrapper>\n </div>\n </div>\n</ng-template>\n", styles: ["@charset \"UTF-8\";.chat-input-wrapper{position:absolute;bottom:0;left:0;right:0;max-width:800px;min-width:350px;margin:0 auto;padding-bottom:40px}.chat-input-wrapper.center{bottom:50%;transform:translateY(50%)}.new-chat-title{padding-bottom:40px;text-align:center;font-weight:500;font-size:24px;color:#000}.new-chat-title img{width:40px;height:32px;margin-right:12px;margin-top:-6px}.chat-input{position:relative;display:flex;flex-direction:column;justify-content:space-between;padding:20px 18px;margin:0 auto;background:#FFFFFF;box-shadow:0 2px 8px #0000001a;border-radius:20px;border:1px solid #D9DADB;z-index:30}.chat-input .input-creative-editor{min-height:60px;max-height:300px;overflow-y:auto;outline:none;box-shadow:none;border:none}.chat-input .input-creative-editor[data-empty=true]:before{content:attr(data-placeholder);color:#9a9b9b}.chat-input .input-bottom{padding-top:16px;display:flex;align-items:center;justify-content:space-between;-webkit-user-select:none;user-select:none}.chat-input .input-bottom .input-tools{display:flex;align-items:center;gap:10px}.chat-input .input-bottom .select-tool{display:flex;align-items:center;justify-content:center;height:36px;padding:0 12px;gap:12px;background:#FFFFFF;border-radius:12px;border:1px solid #DBDCE0;cursor:pointer}.chat-input .input-bottom .select-tool-active{border-color:#b7c8fe;color:#3964fe;background:#edf3fe}.chat-input .input-bottom .send{width:32px;height:32px}::ng-deep .params-select-popover .ant-popover-inner{border-radius:10px}::ng-deep .params-select-popover .select-title{display:flex;align-items:center;justify-content:center;position:relative}::ng-deep .params-select-popover .select-title i{position:absolute;right:0;top:3px;cursor:pointer}::ng-deep .params-select-popover .select-content{max-height:200px;padding:16px 0;overflow-y:auto}.chat-input-backdrop{position:absolute;bottom:0;left:0;width:100%;height:100px;z-index:10;background-image:linear-gradient(to bottom,hsl(0,0%,99%),hsla(0,0%,99%,.8))}.send-loading{min-width:34px;height:34px;border-radius:50%;margin-top:auto;display:flex;flex-shrink:0;align-items:center;flex-direction:column;justify-content:center;cursor:not-allowed;white-space:nowrap;color:#fff;background:#3964fe;transition:background .2s;opacity:.4}.send-loading-inner{width:16px;height:16px}.send-loading svg{will-change:transform;animation:.6s linear infinite send-loading}@keyframes send-loading{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.ɵNzTransitionPatchDirective, selector: "[nz-button], nz-button-group, [nz-icon], [nz-menu-item], [nz-submenu], nz-select-top-control, nz-select-placeholder, nz-input-group", inputs: ["hidden"] }, { kind: "directive", type: i3.NzPopoverDirective, selector: "[nz-popover]", inputs: ["nzPopoverArrowPointAtCenter", "nzPopoverTitle", "nzPopoverContent", "nz-popover", "nzPopoverTrigger", "nzPopoverPlacement", "nzPopoverOrigin", "nzPopoverVisible", "nzPopoverMouseEnterDelay", "nzPopoverMouseLeaveDelay", "nzPopoverOverlayClassName", "nzPopoverOverlayStyle", "nzPopoverBackdrop"], outputs: ["nzPopoverVisibleChange"], exportAs: ["nzPopover"] }, { kind: "component", type: i4.NzRadioComponent, selector: "[nz-radio],[nz-radio-button]", inputs: ["nzValue", "nzDisabled", "nzAutoFocus"], exportAs: ["nzRadio"] }, { kind: "component", type: i4.NzRadioGroupComponent, selector: "nz-radio-group", inputs: ["nzDisabled", "nzButtonStyle", "nzSize", "nzName"], exportAs: ["nzRadioGroup"] }, { kind: "component", type: i5.NzCheckboxComponent, selector: "[nz-checkbox]", inputs: ["nzValue", "nzAutoFocus", "nzDisabled", "nzIndeterminate", "nzChecked", "nzId"], outputs: ["nzCheckedChange"], exportAs: ["nzCheckbox"] }, { kind: "component", type: i5.NzCheckboxWrapperComponent, selector: "nz-checkbox-wrapper", outputs: ["nzOnChange"], exportAs: ["nzCheckboxWrapper"] }, { kind: "directive", type: i6.NzIconDirective, selector: "[nz-icon]", inputs: ["nzSpin", "nzRotate", "nzType", "nzTheme", "nzTwotoneColor", "nzIconfont"], exportAs: ["nzIcon"] }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "pipe", type: i1.AsyncPipe, name: "async" }] });
131
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ChatInputComponent, decorators: [{
132
+ type: Component,
133
+ args: [{ selector: 'ngx-chat-input', template: "<!-- \u65B0\u804A\u5929\u4F4D\u4E8E\u5C4F\u5E55\u4E2D\u95F4\uFF0C\u53D1\u9001\u6D88\u606F\u4E4B\u540E\u4F4D\u4E8E\u5E95\u90E8 -->\n<div class=\"chat-input-wrapper\" #chatInputWrapper [class.center]=\"!messages.length\">\n <div class=\"new-chat-title\" *ngIf=\"!messages.length\">\n <img src=\"/assets/images/logo.png\" />\n <span>\u4F60\u597D\uFF0C\u6709\u4EC0\u4E48\u53EF\u4EE5\u5E2E\u5230\u4F60\uFF1F\u5F00\u59CB\u5BF9\u8BDD\u5427</span>\n </div>\n <div class=\"chat-input\">\n <div\n #editorRef\n class=\"input-creative-editor\"\n contenteditable=\"true\"\n spellcheck=\"false\"\n data-placeholder=\"\u8BF7\u8F93\u5165\u5E76\u53D1\u9001\u6D88\u606F\"\n [attr.data-empty]=\"isEmpty\"\n (input)=\"handleInput()\"\n (keydown.enter)=\"handleEnter($event)\"\n (paste)=\"handlePaste($event)\"\n ></div>\n <div class=\"input-bottom\">\n <div class=\"input-tools\">\n <div class=\"select-tool\" [ngClass]=\"{ 'select-tool-active': deepThink }\" (click)=\"toggleDeepThink()\">\n <svg\n t=\"1764131026949\"\n style=\"width: 14px; height: 14px\"\n class=\"icon\"\n viewBox=\"0 0 1024 1024\"\n version=\"1.1\"\n xmlns=\"http://www.w3.org/2000/svg\"\n p-id=\"5012\"\n width=\"64\"\n height=\"64\"\n >\n <path\n d=\"M903.36 512a795.52 795.52 0 0 1 65.536 136.704c48 132.16 39.616 230.528-25.024 295.168-64.64 64.64-163.008 72.96-295.168 24.96A795.52 795.52 0 0 1 512 903.424a795.52 795.52 0 0 1-136.704 65.536c-132.096 48-230.528 39.616-295.168-25.024-64.64-64.64-72.96-163.008-24.96-295.104 16.64-46.016 38.528-91.52 65.536-136.768A795.52 795.52 0 0 1 55.04 375.232C7.04 243.2 15.36 144.768 80 80.128 144.768 15.488 243.2 7.168 375.296 55.04c46.016 16.64 91.584 38.528 136.768 65.536A795.52 795.52 0 0 1 648.768 55.104c132.096-48 230.464-39.616 295.104 25.024 64.64 64.64 72.96 163.008 25.024 295.104A795.52 795.52 0 0 1 903.36 512z m-53.12-79.424c15.168-28.736 27.968-57.536 38.464-86.4 35.584-98.112 33.92-166.656-5.12-205.696-39.04-39.04-107.648-40.768-205.696-5.12a685.44 685.44 0 0 0-86.4 38.4 1240.96 1240.96 0 0 1 138.432 120.32 1240.832 1240.832 0 0 1 120.32 138.496zM432.576 173.824a685.44 685.44 0 0 0-86.4-38.528c-98.112-35.584-166.656-33.92-205.696 5.12-39.04 39.04-40.768 107.648-5.12 205.696 10.432 28.928 23.296 57.728 38.4 86.4a1240.896 1240.896 0 0 1 120.32-138.432 1240.832 1240.832 0 0 1 138.496-120.32zM173.824 591.488a685.44 685.44 0 0 0-38.464 86.4c-35.648 98.048-33.92 166.592 5.12 205.632 39.04 39.04 107.584 40.768 205.696 5.12a685.44 685.44 0 0 0 86.4-38.4 1240.768 1240.768 0 0 1-138.496-120.32 1240.96 1240.96 0 0 1-120.256-138.432z m495.744 78.08A1112.064 1112.064 0 0 0 802.048 512a1112.064 1112.064 0 0 0-132.48-157.568A1112.128 1112.128 0 0 0 512 221.952a1112.128 1112.128 0 0 0-157.504 132.48A1112.064 1112.064 0 0 0 222.016 512a1112.192 1112.192 0 0 0 132.416 157.568A1112.064 1112.064 0 0 0 512 802.048a1112.128 1112.128 0 0 0 157.568-132.48z m-78.08 180.608c28.672 15.168 57.472 28.032 86.4 38.464 98.048 35.648 166.592 33.92 205.632-5.12 39.04-39.04 40.768-107.584 5.12-205.696a685.504 685.504 0 0 0-38.4-86.4 1240.832 1240.832 0 0 1-120.32 138.496 1240.96 1240.96 0 0 1-138.432 120.32zM585.088 512a73.152 73.152 0 1 1-146.24 0 73.152 73.152 0 0 1 146.304 0z\"\n fill=\"currentColor\"\n p-id=\"5013\"\n ></path>\n </svg>\n \u6DF1\u5EA6\u601D\u8003\n </div>\n <div\n class=\"select-tool\"\n nz-popover\n [nzPopoverContent]=\"modelListContent\"\n nzPopoverTrigger=\"click\"\n nzPopoverOverlayClassName=\"params-select-popover\"\n [nzPopoverVisible]=\"modelPopoverVisible\"\n (nzPopoverVisibleChange)=\"modelVisibleChange($event)\"\n >\n <img src=\"/assets/images/input/toggle-model.png\" style=\"width: 12px; height: 12px\" />\n {{ currentModel ? currentModel?.name : '\u9009\u62E9\u6A21\u578B' }}\n </div>\n <div\n class=\"select-tool\"\n nz-popover\n [nzPopoverContent]=\"knowledgeListContent\"\n nzPopoverTrigger=\"click\"\n nzPopoverOverlayClassName=\"params-select-popover\"\n [nzPopoverVisible]=\"knowledgePopoverVisible\"\n (nzPopoverVisibleChange)=\"knowledgeVisibleChange($event)\"\n >\n <img src=\"/assets/images/input/relate.png\" style=\"width: 14px; height: 8px\" />\n \u5173\u8054\u77E5\u8BC6\u5E93\n </div>\n </div>\n <img class=\"send\" src=\"/assets/images/input/send.png\" (click)=\"send()\" [hidden]=\"sseService.loading$ | async\" />\n <div class=\"send-loading\" *ngIf=\"sseService.loading$ | async\">\n <div class=\"send-loading-inner\">\n <svg viewBox=\"0 0 36 36\" version=\"1.1\" xmlns=\"http://www.w3.org/2000/svg\" aria-hidden=\"true\" data-icon=\"spin\">\n <defs>\n <linearGradient x1=\"0%\" y1=\"100%\" x2=\"100%\" y2=\"100%\" id=\"linearGradient-1\">\n <stop stop-color=\"currentColor\" stop-opacity=\"0\" offset=\"0%\"></stop>\n <stop stop-color=\"currentColor\" stop-opacity=\"0.50\" offset=\"39.9430698%\"></stop>\n <stop stop-color=\"currentColor\" offset=\"100%\"></stop>\n </linearGradient>\n </defs>\n <g stroke=\"none\" stroke-width=\"1\" fill=\"none\" fill-rule=\"evenodd\">\n <rect fill-opacity=\"0.01\" fill=\"none\" x=\"0\" y=\"0\" width=\"36\" height=\"36\"></rect>\n <path\n d=\"M34,18 C34,9.163444 26.836556,2 18,2 C11.6597233,2 6.18078805,5.68784135 3.59122325,11.0354951\"\n stroke=\"url(#linearGradient-1)\"\n stroke-width=\"4\"\n stroke-linecap=\"round\"\n ></path>\n </g>\n </svg>\n </div>\n </div>\n <!-- <img class=\"send\" src=\"/assets/images/input/send-gray.png\" /> -->\n </div>\n </div>\n <div class=\"chat-input-backdrop\"></div>\n</div>\n\n<ng-template #modelListContent>\n <div style=\"width: 200px\">\n <div class=\"select-title\">\n \u5207\u6362\u6A21\u578B\n <i nz-icon nzType=\"close\" style=\"color: #999\" (click)=\"modelVisibleChange(false)\"></i>\n </div>\n <div class=\"select-content\">\n <nz-radio-group [(ngModel)]=\"currentModel\">\n <label\n nz-radio\n [nzValue]=\"item\"\n [style.margin-top.px]=\"i === 0 ? 0 : 10\"\n *ngFor=\"let item of modelList; let i = index\"\n >\n {{ item.name }}\n </label>\n </nz-radio-group>\n </div>\n </div>\n</ng-template>\n\n<ng-template #knowledgeListContent>\n <div style=\"width: 150px\">\n <div class=\"select-title\">\n \u5173\u8054\u77E5\u8BC6\u5E93\n <i nz-icon nzType=\"close\" style=\"color: #999\" (click)=\"knowledgeVisibleChange(false)\"></i>\n </div>\n <div class=\"select-content\">\n <nz-checkbox-wrapper style=\"width: 100%\" (nzOnChange)=\"knowledgeSelectChange($event)\">\n <div *ngFor=\"let item of knowledgeList\">\n <label nz-checkbox [nzValue]=\"item.iCode\">{{ item.name }}</label>\n </div>\n </nz-checkbox-wrapper>\n </div>\n </div>\n</ng-template>\n", styles: ["@charset \"UTF-8\";.chat-input-wrapper{position:absolute;bottom:0;left:0;right:0;max-width:800px;min-width:350px;margin:0 auto;padding-bottom:40px}.chat-input-wrapper.center{bottom:50%;transform:translateY(50%)}.new-chat-title{padding-bottom:40px;text-align:center;font-weight:500;font-size:24px;color:#000}.new-chat-title img{width:40px;height:32px;margin-right:12px;margin-top:-6px}.chat-input{position:relative;display:flex;flex-direction:column;justify-content:space-between;padding:20px 18px;margin:0 auto;background:#FFFFFF;box-shadow:0 2px 8px #0000001a;border-radius:20px;border:1px solid #D9DADB;z-index:30}.chat-input .input-creative-editor{min-height:60px;max-height:300px;overflow-y:auto;outline:none;box-shadow:none;border:none}.chat-input .input-creative-editor[data-empty=true]:before{content:attr(data-placeholder);color:#9a9b9b}.chat-input .input-bottom{padding-top:16px;display:flex;align-items:center;justify-content:space-between;-webkit-user-select:none;user-select:none}.chat-input .input-bottom .input-tools{display:flex;align-items:center;gap:10px}.chat-input .input-bottom .select-tool{display:flex;align-items:center;justify-content:center;height:36px;padding:0 12px;gap:12px;background:#FFFFFF;border-radius:12px;border:1px solid #DBDCE0;cursor:pointer}.chat-input .input-bottom .select-tool-active{border-color:#b7c8fe;color:#3964fe;background:#edf3fe}.chat-input .input-bottom .send{width:32px;height:32px}::ng-deep .params-select-popover .ant-popover-inner{border-radius:10px}::ng-deep .params-select-popover .select-title{display:flex;align-items:center;justify-content:center;position:relative}::ng-deep .params-select-popover .select-title i{position:absolute;right:0;top:3px;cursor:pointer}::ng-deep .params-select-popover .select-content{max-height:200px;padding:16px 0;overflow-y:auto}.chat-input-backdrop{position:absolute;bottom:0;left:0;width:100%;height:100px;z-index:10;background-image:linear-gradient(to bottom,hsl(0,0%,99%),hsla(0,0%,99%,.8))}.send-loading{min-width:34px;height:34px;border-radius:50%;margin-top:auto;display:flex;flex-shrink:0;align-items:center;flex-direction:column;justify-content:center;cursor:not-allowed;white-space:nowrap;color:#fff;background:#3964fe;transition:background .2s;opacity:.4}.send-loading-inner{width:16px;height:16px}.send-loading svg{will-change:transform;animation:.6s linear infinite send-loading}@keyframes send-loading{0%{transform:rotate(0)}to{transform:rotate(360deg)}}\n"] }]
134
+ }], propDecorators: { editor: [{
135
+ type: ViewChild,
136
+ args: ['editorRef']
137
+ }], chatInputWrapper: [{
138
+ type: ViewChild,
139
+ args: ['chatInputWrapper']
140
+ }], messages: [{
141
+ type: Input
142
+ }], sendMessage: [{
143
+ type: Output
144
+ }], messageChange: [{
145
+ type: Output
146
+ }], enter: [{
147
+ type: Output
148
+ }], paste: [{
149
+ type: Output
150
+ }] } });
151
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1pbnB1dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hZ2VudC9jaGF0L2NoYXQtaW5wdXQvY2hhdC1pbnB1dC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hZ2VudC9jaGF0L2NoYXQtaW5wdXQvY2hhdC1pbnB1dC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSwwQkFBMEI7QUFDMUIsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLEVBQUUsWUFBWSxFQUFjLFNBQVMsRUFBRSxNQUFNLEVBQVUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzlHLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUV2RCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFJdEQsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHlCQUF5QixDQUFDOzs7Ozs7Ozs7QUFPckQsTUFBTSxPQUFPLGtCQUFrQjtJQUwvQjtRQVFXLGFBQVEsR0FBYyxFQUFFLENBQUM7UUFDeEIsZ0JBQVcsR0FBRyxJQUFJLFlBQVksRUFBb0QsQ0FBQztRQUNuRixrQkFBYSxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFDM0MsVUFBSyxHQUFHLElBQUksWUFBWSxFQUFVLENBQUM7UUFDbkMsVUFBSyxHQUFHLElBQUksWUFBWSxFQUFTLENBQUM7UUFFcEMsZ0JBQVcsR0FBRyxNQUFNLENBQUMsV0FBVyxDQUFDLENBQUM7UUFDbEMsaUJBQVksR0FBRyxNQUFNLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDckMsZUFBVSxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUV2QyxZQUFPLEdBQUcsSUFBSSxDQUFDO1FBQ2YsaUJBQVksR0FBa0IsSUFBSSxDQUFDO1FBQ25DLHFCQUFnQixHQUFhLEVBQUUsQ0FBQztRQUNoQyxjQUFT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UUFBUSxFQUFFLElBQUk7WUFDZCxTQUFTLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxNQUFNO1lBQ3pDLGFBQWEsRUFBRSxJQUFJLENBQUMsZ0JBQWdCO1NBQ3JDLENBQUM7UUFDRixJQUFJLENBQUMsVUFBVSxFQUFFLENBQUM7UUFDbEIsSUFBSSxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsRUFBRSxPQUFPLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxDQUFDLENBQUM7SUFDMUQsQ0FBQzs7Z0hBbElVLGtCQUFrQjtvR0FBbEIsa0JBQWtCLDZZQ2YvQiw4dk9Ba0lBOzRGRG5IYSxrQkFBa0I7a0JBTDlCLFNBQVM7K0JBQ0UsZ0JBQWdCOzhCQUtGLE1BQU07c0JBQTdCLFNBQVM7dUJBQUMsV0FBVztnQkFDUyxnQkFBZ0I7c0JBQTlDLFNBQVM7dUJBQUMsa0JBQWtCO2dCQUNwQixRQUFRO3NCQUFoQixLQUFLO2dCQUNJLFdBQVc7c0JBQXBCLE1BQU07Z0JBQ0csYUFBYTtzQkFBdEIsTUFBTTtnQkFDRyxLQUFLO3NCQUFkLE1BQU07Z0JBQ0csS0FBSztzQkFBZCxNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiLy8gY2hhdC1pbnB1dC5jb21wb25lbnQudHNcbmltcG9ydCB7IENvbXBvbmVudCwgT3V0cHV0LCBFdmVudEVtaXR0ZXIsIEVsZW1lbnRSZWYsIFZpZXdDaGlsZCwgaW5qZWN0LCBPbkluaXQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBIdHRwU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL2h0dHAuc2VydmljZSc7XG5pbXBvcnQgeyBJTW9kZWwsIElNb2RlbExpc3QgfSBmcm9tICcuLi9tb2RlbHMvbW9kZWwtbGlzdC5tb2RlbCc7XG5pbXBvcnQgeyBTdG9yZVNlcnZpY2UgfSBmcm9tICcuLi9zdG9yZS9zdG9yZS5zZXJ2aWNlJztcbmltcG9ydCB7IElLbm93bGVkZ2VMaXN0IH0gZnJvbSAnLi4vbW9kZWxzL2tub3dsZWRnZS1saXN0Lm1vZGVsJztcbmltcG9ydCB7IElTZW5kTWVzc2FnZSB9IGZyb20gJy4uL21vZGVscy9zZW5kLW1lc3NhZ2UubW9kZWwnO1xuaW1wb3J0IHsgTWVzc2FnZSB9IGZyb20gJy4uL21vZGVscy9tZXNzYWdlLm1vZGVsJztcbmltcG9ydCB7IFNzZVNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9zc2Uuc2VydmljZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ25neC1jaGF0LWlucHV0JyxcbiAgdGVtcGxhdGVVcmw6ICcuL2NoYXQtaW5wdXQuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9jaGF0LWlucHV0LmNvbXBvbmVudC5zY3NzJ10sXG59KVxuZXhwb3J0IGNsYXNzIENoYXRJbnB1dENvbXBvbmVudCBpbXBsZW1lbnRzIE9uSW5pdCB7XG4gIEBWaWV3Q2hpbGQoJ2VkaXRvclJlZicpIGVkaXRvciE6IEVsZW1lbnRSZWY8SFRNTERpdkVsZW1lbnQ+O1xuICBAVmlld0NoaWxkKCdjaGF0SW5wdXRXcmFwcGVyJykgY2hhdElucHV0V3JhcHBlciE6IEVsZW1lbnRSZWY8SFRNTERpdkVsZW1lbnQ+O1xuICBASW5wdXQoKSBtZXNzYWdlczogTWVzc2FnZVtdID0gW107XG4gIEBPdXRwdXQoKSBzZW5kTWVzc2FnZSA9IG5ldyBFdmVudEVtaXR0ZXI8eyBtZXNzYWdlOiBJU2VuZE1lc3NhZ2U7IGlzUmVnZW5lcmF0ZTogYm9vbGVhbiB9PigpO1xuICBAT3V0cHV0KCkgbWVzc2FnZUNoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8c3RyaW5nPigpO1xuICBAT3V0cHV0KCkgZW50ZXIgPSBuZXcgRXZlbnRFbWl0dGVyPHN0cmluZz4oKTtcbiAgQE91dHB1dCgpIHBhc3RlID0gbmV3IEV2ZW50RW1pdHRlcjxFdmVudD4oKTtcblxuICBwcml2YXRlIGh0dHBTZXJ2aWNlID0gaW5qZWN0KEh0dHBTZXJ2aWNlKTtcbiAgcHJpdmF0ZSBzdG9yZVNlcnZpY2UgPSBpbmplY3QoU3RvcmVTZXJ2aWNlKTtcbiAgcHVibGljIHNzZVNlcnZpY2UgPSBpbmplY3QoU3NlU2VydmljZSk7XG5cbiAgaXNFbXB0eSA9IHRydWU7XG4gIGN1cnJlbnRNb2RlbDogSU1vZGVsIHwgbnVsbCA9IG51bGw7XG4gIGN1cnJlbnRLbm93bGVkZ2U6IHN0cmluZ1tdID0gW107XG4gIG1vZGVsTGlzdDogSU1vZGVsTGlzdCA9IFtdO1xuICBrbm93bGVkZ2VMaXN0OiBJS25vd2xlZGdlTGlzdCA9IFtdO1xuICBtb2RlbFBvcG92ZXJWaXNpYmxlID0gZmFsc2U7XG4gIGtub3dsZWRnZVBvcG92ZXJWaXNpYmxlID0gZmFsc2U7XG4gIGRlZXBUaGluayA9IHRydWU7XG5cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgdGhpcy5nZXRNb2RlbExpc3QoKTtcbiAgICB0aGlzLmdldEtub3dsZWRnZUxpc3QoKTtcbiAgfVxuXG4gIC8vIOWPr+S+m+S6pOS6kueahOaooeWei1xuICBnZXRNb2RlbExpc3QoKTogdm9pZCB7XG4gICAgdGhpcy5odHRwU2VydmljZS5nZXQoYCR7dGhpcy5zdG9yZVNlcnZpY2UuZ2V0QmFzZVVybCgpfS9hcGkvQ2hhdE1vZGVsL21vZGVsc2ApLnN1YnNjcmliZSgocmVzOiBJTW9kZWxMaXN0KSA9PiB7XG4gICAgICB0aGlzLm1vZGVsTGlzdCA9IHJlcztcbiAgICAgIGlmIChyZXMubGVuZ3RoKSB7XG4gICAgICAgIHRoaXMuY3VycmVudE1vZGVsID0gcmVzWzBdO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgLy8g5qih5Z6L5Y+v6YCJ55qE55+l6K+G5bqTXG4gIGdldEtub3dsZWRnZUxpc3QoKTogdm9pZCB7XG4gICAgdGhpcy5odHRwU2VydmljZVxuICAgICAgLmdldChgJHt0aGlzLnN0b3JlU2VydmljZS5nZXRCYXNlVXJsKCl9L2FwaS9DaGF0TW9kZWwva25vd2xlZGdlc2ApXG4gICAgICAuc3Vic2NyaWJlKChyZXM6IElLbm93bGVkZ2VMaXN0KSA9PiB7XG4gICAgICAgIHRoaXMua25vd2xlZGdlTGlzdCA9IHJlcztcbiAgICAgIH0pO1xuICB9XG5cbiAgbW9kZWxWaXNpYmxlQ2hhbmdlKHZhbHVlOiBib29sZWFuKTogdm9pZCB7XG4gICAgdGhpcy5tb2RlbFBvcG92ZXJWaXNpYmxlID0gdmFsdWU7XG4gIH1cblxuICBrbm93bGVkZ2VWaXNpYmxlQ2hhbmdlKHZhbHVlOiBib29sZWFuKTogdm9pZCB7XG4gICAgdGhpcy5rbm93bGVkZ2VQb3BvdmVyVmlzaWJsZSA9IHZhbHVlO1xuICB9XG5cbiAga25vd2xlZGdlU2VsZWN0Q2hhbmdlKHZhbHVlOiBzdHJpbmdbXSk6IHZvaWQge1xuICAgIHRoaXMuY3VycmVudEtub3dsZWRnZSA9IHZhbHVlO1xuICB9XG5cbiAgdG9nZ2xlRGVlcFRoaW5rKCk6IHZvaWQge1xuICAgIHRoaXMuZGVlcFRoaW5rID0gIXRoaXMuZGVlcFRoaW5rO1xuICB9XG5cbiAgcHJpdmF0ZSB1cGRhdGVFbXB0eVN0YXRlKCkge1xuICAgIGNvbnN0IGVsID0gdGhpcy5lZGl0b3IubmF0aXZlRWxlbWVudDtcbiAgICBjb25zdCB0ZXh0ID0gZWwudGV4dENvbnRlbnQ/LnRyaW0oKSB8fCAnJztcbiAgICBjb25zdCBoYXNPbmx5QnIgPSBlbC5pbm5lckhUTUwgPT09ICc8YnI+JyB8fCBlbC5pbm5lckhUTUwgPT09ICcnO1xuXG4gICAgdGhpcy5pc0VtcHR5ID0gdGV4dCA9PT0gJycgJiYgaGFzT25seUJyO1xuICB9XG5cbiAgaGFuZGxlSW5wdXQoKTogdm9pZCB7XG4gICAgdGhpcy51cGRhdGVFbXB0eVN0YXRlKCk7XG4gICAgY29uc3QgaHRtbCA9IHRoaXMuZWRpdG9yLm5hdGl2ZUVsZW1lbnQuaW5uZXJIVE1MO1xuICAgIGNvbnN0IHRleHQgPSB0aGlzLmVkaXRvci5uYXRpdmVFbGVtZW50LnRleHRDb250ZW50IHx8ICcnO1xuICAgIC8vIHRoaXMubWVzc2FnZUNoYW5nZS5lbWl0KHRleHQpO1xuICB9XG5cbiAgaGFuZGxlRW50ZXIoZXZlbnQ6IEV2ZW50KTogdm9pZCB7XG4gICAgaWYgKCEoZXZlbnQgYXMgS2V5Ym9hcmRFdmVudCkuc2hpZnRLZXkpIHtcbiAgICAgIGV2ZW50LnByZXZlbnREZWZhdWx0KCk7XG4gICAgICAvLyB0aGlzLmVudGVyLmVtaXQoKGV2ZW50IGFzIGFueSkudGFyZ2V0IS5pbm5lclRleHQhLnJlcGxhY2UoL1xccysvZywgXCIgXCIpLnRyaW0oKSk7XG4gICAgICB0aGlzLnNlbmQoKTtcbiAgICB9XG4gIH1cblxuICBoYW5kbGVQYXN0ZShldmVudDogQ2xpcGJvYXJkRXZlbnQpOiB2b2lkIHtcbiAgICAvLyB0aGlzLnBhc3RlLmVtaXQoZXZlbnQpO1xuICAgIHNldFRpbWVvdXQoKCkgPT4gdGhpcy51cGRhdGVFbXB0eVN0YXRlKCksIDApO1xuICB9XG5cbiAgY2xlYXJJbnB1dCgpOiB2b2lkIHtcbiAgICB0aGlzLmVkaXRvci5uYXRpdmVFbGVtZW50LmlubmVySFRNTCA9ICcnO1xuICAgIHRoaXMuaGFuZGxlSW5wdXQoKTtcbiAgfVxuXG4gIHJlZ2VuZXJhdGVTZW5kKG1zZzogTWVzc2FnZSk6IHZvaWQge1xuICAgIC8vIOato+WcqOWbnuetlOS4rVxuICAgIGlmICh0aGlzLnNzZVNlcnZpY2UubG9hZGluZ1NvdXJjZS52YWx1ZSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjb25zdCBjb252ZXJzYXRpb24gPSB0aGlzLnN0b3JlU2VydmljZS5nZXRDb252ZXJzYXRpb25JQ29kZSgpO1xuICAgIGNvbnN0IG1lc3NhZ2U6IElTZW5kTWVzc2FnZSA9IHtcbiAgICAgIGNvbnZlcnNhdGlvbixcbiAgICAgIHF1ZXN0aW9uSUNvZGU6IG1zZy5pQ29kZSxcbiAgICAgIG1vZGVsOiB0aGlzLmN1cnJlbnRNb2RlbD8uaUNvZGUgPz8gJycsXG4gICAgICB0aGluazogdGhpcy5kZWVwVGhpbmssXG4gICAgICBxdWVzdGlvbjogbXNnLnF1ZXN0aW9uLFxuICAgICAga25vd2xlZGdlOiAhIXRoaXMuY3VycmVudEtub3dsZWRnZS5sZW5ndGgsXG4gICAgICBrbm93bGVkZ2VMaXN0OiB0aGlzLmN1cnJlbnRLbm93bGVkZ2UsXG4gICAgfTtcbiAgICB0aGlzLnNlbmRNZXNzYWdlLmVtaXQoeyBtZXNzYWdlLCBpc1JlZ2VuZXJhdGU6IHRydWUgfSk7XG4gIH1cblxuICBzZW5kKCk6IHZvaWQge1xuICAgIC8vIOato+WcqOWbnuetlOS4rVxuICAgIGlmICh0aGlzLnNzZVNlcnZpY2UubG9hZGluZ1NvdXJjZS52YWx1ZSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjb25zdCB0ZXh0ID0gdGhpcy5lZGl0b3IubmF0aXZlRWxlbWVudC50ZXh0Q29udGVudCB8fCAnJztcbiAgICBjb25zdCBjb252ZXJzYXRpb24gPSB0aGlzLnN0b3JlU2VydmljZS5nZXRDb252ZXJzYXRpb25JQ29kZSgpO1xuICAgIGNvbnN0IG1lc3NhZ2U6IElTZW5kTWVzc2FnZSA9IHtcbiAgICAgIGNvbnZlcnNhdGlvbixcbiAgICAgIG1vZGVsOiB0aGlzLmN1cnJlbnRNb2RlbD8uaUNvZGUgPz8gJycsXG4gICAgICB0aGluazogdGhpcy5kZWVwVGhpbmssXG4gICAgICBxdWVzdGlvbjogdGV4dCxcbiAgICAgIGtub3dsZWRnZTogISF0aGlzLmN1cnJlbnRLbm93bGVkZ2UubGVuZ3RoLFxuICAgICAga25vd2xlZGdlTGlzdDogdGhpcy5jdXJyZW50S25vd2xlZGdlLFxuICAgIH07XG4gICAgdGhpcy5jbGVhcklucHV0KCk7XG4gICAgdGhpcy5zZW5kTWVzc2FnZS5lbWl0KHsgbWVzc2FnZSwgaXNSZWdlbmVyYXRlOiBmYWxzZSB9KTtcbiAgfVxufVxuIiwiPCEtLSDmlrDogYrlpKnkvY3kuo7lsY/luZXkuK3pl7TvvIzlj5HpgIHmtojmga/kuYvlkI7kvY3kuo7lupXpg6ggLS0+XG48ZGl2IGNsYXNzPVwiY2hhdC1pbnB1dC13cmFwcGVyXCIgI2NoYXRJbnB1dFdyYXBwZXIgW2NsYXNzLmNlbnRlcl09XCIhbWVzc2FnZXMubGVuZ3RoXCI+XG4gIDxkaXYgY2xhc3M9XCJuZXctY2hhdC10aXRsZVwiICpuZ0lmPVwiIW1lc3NhZ2VzLmxlbmd0aFwiPlxuICAgIDxpbWcgc3JjPVwiL2Fzc2V0cy9pbWFnZXMvbG9nby5wbmdcIiAvPlxuICAgIDxzcGFuPuS9oOWlve+8jOacieS7gOS5iOWPr+S7peW4ruWIsOS9oO+8n+W8gOWni+WvueivneWQpzwvc3Bhbj5cbiAgPC9kaXY+XG4gIDxkaXYgY2xhc3M9XCJjaGF0LWlucHV0XCI+XG4gICAgPGRpdlxuICAgICAgI2VkaXRvclJlZlxuICAgICAgY2xhc3M9XCJpbnB1dC1jcmVhdGl2ZS1lZGl0b3JcIlxuICAgICAgY29udGVudGVkaXRhYmxlPVwidHJ1ZVwiXG4gICAgICBzcGVsbGNoZWNrPVwiZmFsc2VcIlxuICAgICAgZGF0YS1wbGFjZWhvbGRlcj1cIuivt+i+k+WFpeW5tuWPkemAgea2iOaBr1wiXG4gICAgICBbYXR0ci5kYXRhLWVtcHR5XT1cImlzRW1wdHlcIlxuICAgICAgKGlucHV0KT1cImhhbmRsZUlucHV0KClcIlxuICAgICAgKGtleWRvd24uZW50ZXIpPVwiaGFuZGxlRW50ZXIoJGV2ZW50KVwiXG4gICAgICAocGFzdGUpPVwiaGFuZGxlUGFzdGUoJGV2ZW50KVwiXG4gICAgPjwvZGl2PlxuICAgIDxkaXYgY2xhc3M9XCJpbnB1dC1ib3R0b21cIj5cbiAgICAgIDxkaXYgY2xhc3M9XCJpbnB1dC10b29sc1wiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2VsZWN0LXRvb2xcIiBbbmdDbGFzc109XCJ7ICdzZWxlY3QtdG9vbC1hY3RpdmUnOiBkZWVwVGhpbmsgfVwiIChjbGljayk9XCJ0b2dnbGVEZWVwVGhpbmsoKVwiPlxuICAgICAgICAgIDxzdmdcbiAgICAgICAgICAgIHQ9XCIxNzY0MTMxMDI2OTQ5XCJcbiAgICAgICAgICAgIHN0eWxlPVwid2lkdGg6IDE0cHg7IGhlaWdodDogMTRweFwiXG4gICAgICAgICAgICBjbGFzcz1cImljb25cIlxuICAgICAgICAgICAgdmlld0JveD1cIjAgMCAxMDI0IDEwMjRcIlxuICAgICAgICAgICAgdmVyc2lvbj1cIjEuMVwiXG4gICAgICAgICAgICB4bWxucz1cImh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnXCJcbiAgICAgICAgICAgIHAtaWQ9XCI1MDEyXCJcbiAgICAgICAgICAgIHdpZHRoPVwiNjRcIlxuICAgICAgICAgICAgaGVpZ2h0PVwiNjRcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIDxwYXRoXG4gICAgICAgICAgICAgIGQ9XCJNOTAzLjM2IDUxMmE3OTUuNTIgNzk1LjUyIDAgMCAxIDY1LjUzNiAxMzYuNzA0YzQ4IDEzMi4xNiAzOS42MTYgMjMwLjUyOC0yNS4wMjQgMjk1LjE2OC02NC42NCA2NC42NC0xNjMuMDA4IDcyLjk2LTI5NS4xNjggMjQuOTZBNzk1LjUyIDc5NS41MiAwIDAgMSA1MTIgOTAzLjQyNGE3OTUuNTIgNzk1LjUyIDAgMCAxLTEzNi43MDQgNjUuNTM2Yy0xMzIuMDk2IDQ4LTIzMC41MjggMzkuNjE2LTI5NS4xNjgtMjUuMDI0LTY0LjY0LTY0LjY0LTcyLjk2LTE2My4wMDgtMjQuOTYtMjk1LjEwNCAxNi42NC00Ni4wMTYgMzguNTI4LTkxLjUyIDY1LjUzNi0xMzYuNzY4QTc5NS41MiA3OTUuNTIgMCAwIDEgNTUuMDQgMzc1LjIzMkM3LjA0IDI0My4yIDE1LjM2IDE0NC43NjggODAgODAuMTI4IDE0NC43NjggMTUuNDg4IDI0My4yIDcuMTY4IDM3NS4yOTYgNTUuMDRjNDYuMDE2IDE2LjY0IDkxLjU4NCAzOC41MjggMTM2Ljc2OCA2NS41MzZBNzk1LjUyIDc5NS41MiAwIDAgMSA2NDguNzY4IDU1LjEwNGMxMzIuMDk2LTQ4IDIzMC40NjQtMzkuNjE2IDI5NS4xMDQgMjUuMDI0IDY0LjY0IDY0LjY0IDcyLjk2IDE2My4wMDggMjUuMDI0IDI5NS4xMDRBNzk1LjUyIDc5NS41MiAwIDAgMSA5MDMuMzYgNTEyeiBtLTUzLjEyLTc5LjQyNGMxNS4xNjgtMjguNzM2IDI3Ljk2OC01Ny41MzYgMzguNDY0LTg2LjQgMzUuNTg0LTk4LjExMiAzMy45Mi0xNjYuNjU2LTUuMTItMjA1LjY5Ni0zOS4wNC0zOS4wNC0xMDcuNjQ4LTQwLjc2OC0yMDUuNjk2LTUuMTJhNjg1LjQ0IDY4NS40NCAwIDAgMC04Ni40IDM4LjQgMTI0MC45NiAxMjQwLjk2IDAgMCAxIDEzOC40MzIgMTIwLjMyIDEyNDAuODMyIDEyNDAuODMyIDAgMCAxIDEyMC4zMiAxMzguNDk2ek00MzIuNTc2IDE3My44MjRhNjg1LjQ0IDY4NS40NCAwIDAgMC04Ni40LTM4LjUyOGMtOTguMTEyLTM1LjU4NC0xNjYuNjU2LTMzLjkyLTIwNS42OTYgNS4xMi0zOS4wNCAzOS4wNC00MC43NjggMTA3LjY0OC01LjEyIDIwNS42OTYgMTAuNDMyIDI4LjkyOCAyMy4yOTYgNTcuNzI4IDM4LjQgODYuNGExMjQwLjg5NiAxMjQwLjg5NiAwIDAgMSAxMjAuMzItMTM4LjQzMiAxMjQwLjgzMiAxMjQwLjgzMiAwIDAgMSAxMzguNDk2LTEyMC4zMnpNMTczLjgyNCA1OTEuNDg4YTY4NS40NCA2ODUuNDQgMCAwIDAtMzguNDY0IDg2LjRjLTM1LjY0OCA5OC4wNDgtMzMuOTIgMTY2LjU5MiA1LjEyIDIwNS42MzIgMzkuMDQgMzkuMDQgMTA3LjU4NCA0MC43NjggMjA1LjY5NiA1LjEyYTY4NS40NCA2ODUuNDQgMCAwIDAgODYuNC0zOC40IDEyNDAuNzY4IDEyNDAuNzY4IDAgMCAxLTEzOC40OTYtMTIwLjMyIDEyNDAuOTYgMTI0MC45NiAwIDAgMS0xMjAuMjU2LTEzOC40MzJ6IG00OTUuNzQ0IDc4LjA4QTExMTIuMDY0IDExMTIuMDY0IDAgMCAwIDgwMi4wNDggNTEyYTExMTIuMDY0IDExMTIuMDY0IDAgMCAwLTEzMi40OC0xNTcuNTY4QTExMTIuMTI4IDExMTIuMTI4IDAgMCAwIDUxMiAyMjEuOTUyYTExMTIuMTI4IDExMTIuMTI4IDAgMCAwLTE1Ny41MDQgMTMyLjQ4QTExMTIuMDY0IDExMTIuMDY0IDAgMCAwIDIyMi4wMTYgNTEyYTExMTIuMTkyIDExMTIuMTkyIDAgMCAwIDEzMi40MTYgMTU3LjU2OEExMTEyLjA2NCAxMTEyLjA2NCAwIDAgMCA1MTIgODAyLjA0OGExMTEyLjEyOCAxMTEyLjEyOCAwIDAgMCAxNTcuNTY4LTEzMi40OHogbS03OC4wOCAxODAuNjA4YzI4LjY3MiAxNS4xNjggNTcuNDcyIDI4LjAzMiA4Ni40IDM4LjQ2NCA5OC4wNDggMzUuNjQ4IDE2Ni41OTIgMzMuOTIgMjA1LjYzMi01LjEyIDM5LjA0LTM5LjA0IDQwLjc2OC0xMDcuNTg0IDUuMTItMjA1LjY5NmE2ODUuNTA0IDY4NS41MDQgMCAwIDAtMzguNC04Ni40IDEyNDAuODMyIDEyNDAuODMyIDAgMCAxLTEyMC4zMiAxMzguNDk2IDEyNDAuOTYgMTI0MC45NiAwIDAgMS0xMzguNDMyIDEyMC4zMnpNNTg1LjA4OCA1MTJhNzMuMTUyIDczLjE1MiAwIDEgMS0xNDYuMjQgMCA3My4xNTIgNzMuMTUyIDAgMCAxIDE0Ni4zMDQgMHpcIlxuICAgICAgICAgICAgICBmaWxsPVwiY3VycmVudENvbG9yXCJcbiAgICAgICAgICAgICAgcC1pZD1cIjUwMTNcIlxuICAgICAgICAgICAgPjwvcGF0aD5cbiAgICAgICAgICA8L3N2Zz5cbiAgICAgICAgICDmt7HluqbmgJ3ogINcbiAgICAgICAgPC9kaXY+XG4gICAgICAgIDxkaXZcbiAgICAgICAgICBjbGFzcz1cInNlbGVjdC10b29sXCJcbiAgICAgICAgICBuei1wb3BvdmVyXG4gICAgICAgICAgW256UG9wb3ZlckNvbnRlbnRdPVwibW9kZWxMaXN0Q29udGVudFwiXG4gICAgICAgICAgbnpQb3BvdmVyVHJpZ2dlcj1cImNsaWNrXCJcbiAgICAgICAgICBuelBvcG92ZXJPdmVybGF5Q2xhc3NOYW1lPVwicGFyYW1zLXNlbGVjdC1wb3BvdmVyXCJcbiAgICAgICAgICBbbnpQb3BvdmVyVmlzaWJsZV09XCJtb2RlbFBvcG92ZXJWaXNpYmxlXCJcbiAgICAgICAgICAobnpQb3BvdmVyVmlzaWJsZUNoYW5nZSk9XCJtb2RlbFZpc2libGVDaGFuZ2UoJGV2ZW50KVwiXG4gICAgICAgID5cbiAgICAgICAgICA8aW1nIHNyYz1cIi9hc3NldHMvaW1hZ2VzL2lucHV0L3RvZ2dsZS1tb2RlbC5wbmdcIiBzdHlsZT1cIndpZHRoOiAxMnB4OyBoZWlnaHQ6IDEycHhcIiAvPlxuICAgICAgICAgIHt7IGN1cnJlbnRNb2RlbCA/IGN1cnJlbnRNb2RlbD8ubmFtZSA6ICfpgInmi6nmqKHlnosnIH19XG4gICAgICAgIDwvZGl2PlxuICAgICAgICA8ZGl2XG4gICAgICAgICAgY2xhc3M9XCJzZWxlY3QtdG9vbFwiXG4gICAgICAgICAgbnotcG9wb3ZlclxuICAgICAgICAgIFtuelBvcG92ZXJDb250ZW50XT1cImtub3dsZWRnZUxpc3RDb250ZW50XCJcbiAgICAgICAgICBuelBvcG92ZXJUcmlnZ2VyPVwiY2xpY2tcIlxuICAgICAgICAgIG56UG9wb3Zlck92ZXJsYXlDbGFzc05hbWU9XCJwYXJhbXMtc2VsZWN0LXBvcG92ZXJcIlxuICAgICAgICAgIFtuelBvcG92ZXJWaXNpYmxlXT1cImtub3dsZWRnZVBvcG92ZXJWaXNpYmxlXCJcbiAgICAgICAgICAobnpQb3BvdmVyVmlzaWJsZUNoYW5nZSk9XCJrbm93bGVkZ2VWaXNpYmxlQ2hhbmdlKCRldmVudClcIlxuICAgICAgICA+XG4gICAgICAgICAgPGltZyBzcmM9XCIvYXNzZXRzL2ltYWdlcy9pbnB1dC9yZWxhdGUucG5nXCIgc3R5bGU9XCJ3aWR0aDogMTRweDsgaGVpZ2h0OiA4cHhcIiAvPlxuICAgICAgICAgIOWFs+iBlOefpeivhuW6k1xuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvZGl2PlxuICAgICAgPGltZyBjbGFzcz1cInNlbmRcIiBzcmM9XCIvYXNzZXRzL2ltYWdlcy9pbnB1dC9zZW5kLnBuZ1wiIChjbGljayk9XCJzZW5kKClcIiBbaGlkZGVuXT1cInNzZVNlcnZpY2UubG9hZGluZyQgfCBhc3luY1wiIC8+XG4gICAgICA8ZGl2IGNsYXNzPVwic2VuZC1sb2FkaW5nXCIgKm5nSWY9XCJzc2VTZXJ2aWNlLmxvYWRpbmckIHwgYXN5bmNcIj5cbiAgICAgICAgPGRpdiBjbGFzcz1cInNlbmQtbG9hZGluZy1pbm5lclwiPlxuICAgICAgICAgIDxzdmcgdmlld0JveD1cIjAgMCAzNiAzNlwiIHZlcnNpb249XCIxLjFcIiB4bWxucz1cImh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnXCIgYXJpYS1oaWRkZW49XCJ0cnVlXCIgZGF0YS1pY29uPVwic3BpblwiPlxuICAgICAgICAgICAgPGRlZnM+XG4gICAgICAgICAgICAgIDxsaW5lYXJHcmFkaWVudCB4MT1cIjAlXCIgeTE9XCIxMDAlXCIgeDI9XCIxMDAlXCIgeTI9XCIxMDAlXCIgaWQ9XCJsaW5lYXJHcmFkaWVudC0xXCI+XG4gICAgICAgICAgICAgICAgPHN0b3Agc3RvcC1jb2xvcj1cImN1cnJlbnRDb2xvclwiIHN0b3Atb3BhY2l0eT1cIjBcIiBvZmZzZXQ9XCIwJVwiPjwvc3RvcD5cbiAgICAgICAgICAgICAgICA8c3RvcCBzdG9wLWNvbG9yPVwiY3VycmVudENvbG9yXCIgc3RvcC1vcGFjaXR5PVwiMC41MFwiIG9mZnNldD1cIjM5Ljk0MzA2OTglXCI+PC9zdG9wPlxuICAgICAgICAgICAgICAgIDxzdG9wIHN0b3AtY29sb3I9XCJjdXJyZW50Q29sb3JcIiBvZmZzZXQ9XCIxMDAlXCI+PC9zdG9wPlxuICAgICAgICAgICAgICA8L2xpbmVhckdyYWRpZW50PlxuICAgICAgICAgICAgPC9kZWZzPlxuICAgICAgICAgICAgPGcgc3Ryb2tlPVwibm9uZVwiIHN0cm9rZS13aWR0aD1cIjFcIiBmaWxsPVwibm9uZVwiIGZpbGwtcnVsZT1cImV2ZW5vZGRcIj5cbiAgICAgICAgICAgICAgPHJlY3QgZmlsbC1vcGFjaXR5PVwiMC4wMVwiIGZpbGw9XCJub25lXCIgeD1cIjBcIiB5PVwiMFwiIHdpZHRoPVwiMzZcIiBoZWlnaHQ9XCIzNlwiPjwvcmVjdD5cbiAgICAgICAgICAgICAgPHBhdGhcbiAgICAgICAgICAgICAgICBkPVwiTTM0LDE4IEMzNCw5LjE2MzQ0NCAyNi44MzY1NTYsMiAxOCwyIEMxMS42NTk3MjMzLDIgNi4xODA3ODgwNSw1LjY4Nzg0MTM1IDMuNTkxMjIzMjUsMTEuMDM1NDk1MVwiXG4gICAgICAgICAgICAgICAgc3Ryb2tlPVwidXJsKCNsaW5lYXJHcmFkaWVudC0xKVwiXG4gICAgICAgICAgICAgICAgc3Ryb2tlLXdpZHRoPVwiNFwiXG4gICAgICAgICAgICAgICAgc3Ryb2tlLWxpbmVjYXA9XCJyb3VuZFwiXG4gICAgICAgICAgICAgID48L3BhdGg+XG4gICAgICAgICAgICA8L2c+XG4gICAgICAgICAgPC9zdmc+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgICA8IS0tIDxpbWcgY2xhc3M9XCJzZW5kXCIgc3JjPVwiL2Fzc2V0cy9pbWFnZXMvaW5wdXQvc2VuZC1ncmF5LnBuZ1wiIC8+IC0tPlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbiAgPGRpdiBjbGFzcz1cImNoYXQtaW5wdXQtYmFja2Ryb3BcIj48L2Rpdj5cbjwvZGl2PlxuXG48bmctdGVtcGxhdGUgI21vZGVsTGlzdENvbnRlbnQ+XG4gIDxkaXYgc3R5bGU9XCJ3aWR0aDogMjAwcHhcIj5cbiAgICA8ZGl2IGNsYXNzPVwic2VsZWN0LXRpdGxlXCI+XG4gICAgICDliIfmjaLmqKHlnotcbiAgICAgIDxpIG56LWljb24gbnpUeXBlPVwiY2xvc2VcIiBzdHlsZT1cImNvbG9yOiAjOTk5XCIgKGNsaWNrKT1cIm1vZGVsVmlzaWJsZUNoYW5nZShmYWxzZSlcIj48L2k+XG4gICAgPC9kaXY+XG4gICAgPGRpdiBjbGFzcz1cInNlbGVjdC1jb250ZW50XCI+XG4gICAgICA8bnotcmFkaW8tZ3JvdXAgWyhuZ01vZGVsKV09XCJjdXJyZW50TW9kZWxcIj5cbiAgICAgICAgPGxhYmVsXG4gICAgICAgICAgbnotcmFkaW9cbiAgICAgICAgICBbbnpWYWx1ZV09XCJpdGVtXCJcbiAgICAgICAgICBbc3R5bGUubWFyZ2luLXRvcC5weF09XCJpID09PSAwID8gMCA6IDEwXCJcbiAgICAgICAgICAqbmdGb3I9XCJsZXQgaXRlbSBvZiBtb2RlbExpc3Q7IGxldCBpID0gaW5kZXhcIlxuICAgICAgICA+XG4gICAgICAgICAge3sgaXRlbS5uYW1lIH19XG4gICAgICAgIDwvbGFiZWw+XG4gICAgICA8L256LXJhZGlvLWdyb3VwPlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbjwvbmctdGVtcGxhdGU+XG5cbjxuZy10ZW1wbGF0ZSAja25vd2xlZGdlTGlzdENvbnRlbnQ+XG4gIDxkaXYgc3R5bGU9XCJ3aWR0aDogMTUwcHhcIj5cbiAgICA8ZGl2IGNsYXNzPVwic2VsZWN0LXRpdGxlXCI+XG4gICAgICDlhbPogZTnn6Xor4blupNcbiAgICAgIDxpIG56LWljb24gbnpUeXBlPVwiY2xvc2VcIiBzdHlsZT1cImNvbG9yOiAjOTk5XCIgKGNsaWNrKT1cImtub3dsZWRnZVZpc2libGVDaGFuZ2UoZmFsc2UpXCI+PC9pPlxuICAgIDwvZGl2PlxuICAgIDxkaXYgY2xhc3M9XCJzZWxlY3QtY29udGVudFwiPlxuICAgICAgPG56LWNoZWNrYm94LXdyYXBwZXIgc3R5bGU9XCJ3aWR0aDogMTAwJVwiIChuek9uQ2hhbmdlKT1cImtub3dsZWRnZVNlbGVjdENoYW5nZSgkZXZlbnQpXCI+XG4gICAgICAgIDxkaXYgKm5nRm9yPVwibGV0IGl0ZW0gb2Yga25vd2xlZGdlTGlzdFwiPlxuICAgICAgICAgIDxsYWJlbCBuei1jaGVja2JveCBbbnpWYWx1ZV09XCJpdGVtLmlDb2RlXCI+e3sgaXRlbS5uYW1lIH19PC9sYWJlbD5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L256LWNoZWNrYm94LXdyYXBwZXI+XG4gICAgPC9kaXY+XG4gIDwvZGl2PlxuPC9uZy10ZW1wbGF0ZT5cbiJdfQ==
@@ -0,0 +1,110 @@
1
+ // chat-messages.component.ts
2
+ import { Component, Input, ViewChild, inject, Output, EventEmitter, } from '@angular/core';
3
+ import { MarkdownStreamService } from '../services/markdown-stream.service';
4
+ import { SseService } from '../services/sse.service';
5
+ import { DomSanitizer } from '@angular/platform-browser';
6
+ import { NzMessageService } from 'ng-zorro-antd/message';
7
+ import * as i0 from "@angular/core";
8
+ import * as i1 from "@angular/common";
9
+ import * as i2 from "../components/pagination/pagination.component";
10
+ import * as i3 from "../pipes/markdown.pipe";
11
+ export class ChatMessagesComponent {
12
+ constructor() {
13
+ this.messages = [];
14
+ /**
15
+ * 重新生成回答
16
+ */
17
+ this.regenerateAnswer = new EventEmitter();
18
+ this.md = inject(MarkdownStreamService);
19
+ this.sseService = inject(SseService);
20
+ this.sanitizer = inject(DomSanitizer);
21
+ this.msg = inject(NzMessageService);
22
+ this.isLoading = false;
23
+ this.inputHeight = 180;
24
+ this.renderedHtml = '';
25
+ this.accumulatedMarkdown = '';
26
+ }
27
+ ngAfterViewInit() {
28
+ this.sseService.loading$.subscribe(status => {
29
+ this.isLoading = status;
30
+ });
31
+ this.sseService.delta$.subscribe(delta => {
32
+ if (!delta)
33
+ return;
34
+ this.accumulatedMarkdown += delta;
35
+ this.renderedHtml = this.md.renderIncremental(this.accumulatedMarkdown);
36
+ });
37
+ }
38
+ ngAfterViewChecked() {
39
+ // 正在回答中
40
+ if (this.isLoading) {
41
+ this.scrollToBottom();
42
+ }
43
+ }
44
+ ngOnChanges(changes) {
45
+ if (changes['messages']) {
46
+ setTimeout(() => {
47
+ this.scrollToBottom();
48
+ }, 50);
49
+ }
50
+ }
51
+ // 滚动到底部
52
+ scrollToBottom() {
53
+ if (!this.messagesContainer) {
54
+ return;
55
+ }
56
+ this.messagesContainer.nativeElement.scrollTop = this.messagesContainer.nativeElement.scrollHeight;
57
+ }
58
+ copyContent(text) {
59
+ if (navigator.clipboard && window.isSecureContext) {
60
+ // 使用现代 Clipboard API
61
+ navigator.clipboard.writeText(text).then(() => {
62
+ this.msg.success('复制成功');
63
+ });
64
+ }
65
+ }
66
+ onPageChange(num, msg) {
67
+ msg.answerIndex = num - 1;
68
+ }
69
+ /**
70
+ * 重新生成多个回答
71
+ */
72
+ regenerate(msg) {
73
+ // const question = this.messages[index - 1].message;
74
+ // 重新生成使用底部最新的配置,重新调用sendMessage
75
+ this.regenerateAnswer.emit(msg);
76
+ }
77
+ // 把完整的 markdown 块渲染,剩下的留给打字区
78
+ renderCompleteBlocks(markdown) {
79
+ // 找到最后一个换行符之前的都渲染(足够丝滑)
80
+ const lines = markdown.split('\n');
81
+ let breakPoint = lines.length;
82
+ // 从后往前找,保留未完成的代码块/表格/列表
83
+ for (let i = lines.length - 1; i >= 0; i--) {
84
+ const line = lines[i];
85
+ if (line.trim() === '' || line.startsWith('```') || line.startsWith('|') || line.startsWith('- ')) {
86
+ breakPoint = i;
87
+ break;
88
+ }
89
+ }
90
+ const completePart = lines.slice(0, breakPoint).join('\n');
91
+ const remainingPart = lines.slice(breakPoint).join('\n');
92
+ const html = this.md.renderIncremental(completePart);
93
+ // const html = marked(completePart, { breaks: true, gfm: true });
94
+ return { rendered: html, remaining: remainingPart };
95
+ }
96
+ }
97
+ ChatMessagesComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ChatMessagesComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
98
+ ChatMessagesComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: ChatMessagesComponent, selector: "ngx-chat-messages", inputs: { messages: "messages" }, outputs: { regenerateAnswer: "regenerateAnswer" }, viewQueries: [{ propertyName: "messagesContainer", first: true, predicate: ["messagesContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<!-- chat-messages.component.html -->\n<script src=\"chat-messages.component.ts\"></script>\n<div class=\"messages-container\" #messagesContainer>\n <div class=\"message-list\" [style.--input-height.px]=\"inputHeight\">\n <div class=\"question message-wrapper\" *ngFor=\"let msg of messages; let i = index\">\n <!-- Q -->\n <div class=\"bubble-wrapper user-msg\">\n <div class=\"message-bubble\">\n <div class=\"message-content message-human\">\n {{ msg.question }}\n </div>\n </div>\n </div>\n <!-- A -->\n <div class=\"bubble-wrapper ai-msg\" *ngIf=\"msg.answers[msg.answerIndex]\">\n <div\n class=\"bubble-loading\"\n *ngIf=\"!msg.answers[msg.answerIndex].think && !msg.answers[msg.answerIndex].content\"\n >\n <div class=\"bubble-loading-dot\" *ngFor=\"let item of [1, 2, 3]\"></div>\n </div>\n <div class=\"message-bubble\">\n <!-- \u601D\u8003 -->\n <div class=\"message-think\" *ngIf=\"msg.answers[msg.answerIndex].think\">\n <div\n class=\"message-think-header\"\n (click)=\"msg.answers[msg.answerIndex].thinkExpand = !msg.answers[msg.answerIndex].thinkExpand\"\n >\n <div class=\"message-think-header-title\">\n <img src=\"/assets/images/card/think-finish.png\" />\n {{ msg.answers[msg.answerIndex].content ? '\u5DF2\u5B8C\u6210\u601D\u8003' : '\u6B63\u5728\u601D\u8003\u4E2D...' }}\n </div>\n <svg width=\"18\" height=\"18\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\">\n <polyline points=\"6 9 12 15 18 9\"></polyline>\n </svg>\n </div>\n <div\n class=\"message-content message-assistant message-markdown\"\n [innerHTML]=\"msg.answers[msg.answerIndex].think | markdown\"\n [hidden]=\"!msg.answers[msg.answerIndex].thinkExpand\"\n ></div>\n </div>\n <!-- \u56DE\u7B54 -->\n <div\n class=\"message-content message-assistant message-markdown\"\n [innerHTML]=\"msg.answers[msg.answerIndex].content | markdown\"\n ></div>\n <!-- \u56DE\u7B54\u5DE5\u5177\u680F -->\n <div class=\"assistant-tools\" *ngIf=\"msg.answers[msg.answerIndex].content\">\n <ngx-simple-pagination\n [current]=\"msg.answerIndex + 1\"\n [total]=\"msg.answers.length\"\n (pageChange)=\"onPageChange($event, msg)\"\n *ngIf=\"msg.answers.length > 1\"\n ></ngx-simple-pagination>\n <img\n src=\"assets/images/card/copy.png\"\n alt=\"\u590D\u5236\"\n (click)=\"copyContent(msg.answers[msg.answerIndex].content)\"\n />\n <img\n src=\"assets/images/card/refresh.png\"\n alt=\"\u91CD\u65B0\u751F\u6210\"\n (click)=\"regenerate(msg)\"\n *ngIf=\"i === messages.length - 1\"\n />\n </div>\n </div>\n </div>\n </div>\n\n <!-- <div style=\"padding-bottom: 160px\"></div> -->\n </div>\n</div>\n", styles: [":host{width:100%;height:100%}.messages-container{width:100%;height:100%;overflow-y:auto}.message-list{height:auto;padding-top:16px;padding-bottom:var(--input-height, 210px);display:flex;flex-direction:column;align-items:center;justify-content:center;font-size:16px}.message-wrapper{display:flex;flex-direction:column;justify-content:center;align-items:center;gap:16px;width:100%;margin-bottom:16px;max-width:800px;min-width:350px}.bubble-wrapper{width:100%;display:flex;flex-direction:column;justify-content:center}.bubble-wrapper.user-msg{align-items:end}.bubble-wrapper.ai-msg{align-items:start}.bubble-loading{display:flex;align-items:center;justify-content:flex-start;height:24px;margin-top:4px;margin-bottom:16px}.bubble-loading-dot{width:6px;height:6px;border-radius:50%;margin-right:6px;background-color:#bbb;animation:.9s linear infinite loading-dot}.bubble-loading-dot:first-child{animation-delay:0s}.bubble-loading-dot:nth-child(2){animation-delay:.1s}.bubble-loading-dot:nth-child(3){animation-delay:.2s}@keyframes loading-dot{0%,to{opacity:.3}33%,66%{opacity:1}}.message-bubble{width:100%;display:flex;align-items:flex-end;flex-direction:column;gap:8px}.message-bubble .message-content.message-human{padding:8px 16px;margin-left:auto;border-radius:12px;background:#E4EDFD;color:#004ad3}.message-bubble .message-content.message-assistant{padding:0;width:100%;background:#fff}.message-bubble .message-markdown{background:#fff}.message-bubble .message-think{width:100%;padding:12px;background:#FFFFFF;border-radius:10px;border:1px solid #D9DADB;font-size:14px}.message-bubble .message-think-header{display:flex;align-items:center;justify-content:space-between;cursor:pointer;color:#000;font-size:16px}.message-bubble .message-think-header-title{display:flex;align-items:center;gap:10px}.message-bubble .message-think-header-title img{width:14px;height:14px}.message-bubble .message-think .message-content{padding-top:12px;color:#61666b}.message-bubble .message-time{font-size:12px;color:#999}.assistant-tools{width:100%;display:flex;align-items:center;gap:24px}.assistant-tools img{width:14px;height:14px;cursor:pointer}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.SimplePaginationComponent, selector: "ngx-simple-pagination", inputs: ["current", "total"], outputs: ["pageChange"] }, { kind: "pipe", type: i3.MarkdownPipe, name: "markdown" }] });
99
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ChatMessagesComponent, decorators: [{
100
+ type: Component,
101
+ args: [{ selector: 'ngx-chat-messages', template: "<!-- chat-messages.component.html -->\n<script src=\"chat-messages.component.ts\"></script>\n<div class=\"messages-container\" #messagesContainer>\n <div class=\"message-list\" [style.--input-height.px]=\"inputHeight\">\n <div class=\"question message-wrapper\" *ngFor=\"let msg of messages; let i = index\">\n <!-- Q -->\n <div class=\"bubble-wrapper user-msg\">\n <div class=\"message-bubble\">\n <div class=\"message-content message-human\">\n {{ msg.question }}\n </div>\n </div>\n </div>\n <!-- A -->\n <div class=\"bubble-wrapper ai-msg\" *ngIf=\"msg.answers[msg.answerIndex]\">\n <div\n class=\"bubble-loading\"\n *ngIf=\"!msg.answers[msg.answerIndex].think && !msg.answers[msg.answerIndex].content\"\n >\n <div class=\"bubble-loading-dot\" *ngFor=\"let item of [1, 2, 3]\"></div>\n </div>\n <div class=\"message-bubble\">\n <!-- \u601D\u8003 -->\n <div class=\"message-think\" *ngIf=\"msg.answers[msg.answerIndex].think\">\n <div\n class=\"message-think-header\"\n (click)=\"msg.answers[msg.answerIndex].thinkExpand = !msg.answers[msg.answerIndex].thinkExpand\"\n >\n <div class=\"message-think-header-title\">\n <img src=\"/assets/images/card/think-finish.png\" />\n {{ msg.answers[msg.answerIndex].content ? '\u5DF2\u5B8C\u6210\u601D\u8003' : '\u6B63\u5728\u601D\u8003\u4E2D...' }}\n </div>\n <svg width=\"18\" height=\"18\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\">\n <polyline points=\"6 9 12 15 18 9\"></polyline>\n </svg>\n </div>\n <div\n class=\"message-content message-assistant message-markdown\"\n [innerHTML]=\"msg.answers[msg.answerIndex].think | markdown\"\n [hidden]=\"!msg.answers[msg.answerIndex].thinkExpand\"\n ></div>\n </div>\n <!-- \u56DE\u7B54 -->\n <div\n class=\"message-content message-assistant message-markdown\"\n [innerHTML]=\"msg.answers[msg.answerIndex].content | markdown\"\n ></div>\n <!-- \u56DE\u7B54\u5DE5\u5177\u680F -->\n <div class=\"assistant-tools\" *ngIf=\"msg.answers[msg.answerIndex].content\">\n <ngx-simple-pagination\n [current]=\"msg.answerIndex + 1\"\n [total]=\"msg.answers.length\"\n (pageChange)=\"onPageChange($event, msg)\"\n *ngIf=\"msg.answers.length > 1\"\n ></ngx-simple-pagination>\n <img\n src=\"assets/images/card/copy.png\"\n alt=\"\u590D\u5236\"\n (click)=\"copyContent(msg.answers[msg.answerIndex].content)\"\n />\n <img\n src=\"assets/images/card/refresh.png\"\n alt=\"\u91CD\u65B0\u751F\u6210\"\n (click)=\"regenerate(msg)\"\n *ngIf=\"i === messages.length - 1\"\n />\n </div>\n </div>\n </div>\n </div>\n\n <!-- <div style=\"padding-bottom: 160px\"></div> -->\n </div>\n</div>\n", styles: [":host{width:100%;height:100%}.messages-container{width:100%;height:100%;overflow-y:auto}.message-list{height:auto;padding-top:16px;padding-bottom:var(--input-height, 210px);display:flex;flex-direction:column;align-items:center;justify-content:center;font-size:16px}.message-wrapper{display:flex;flex-direction:column;justify-content:center;align-items:center;gap:16px;width:100%;margin-bottom:16px;max-width:800px;min-width:350px}.bubble-wrapper{width:100%;display:flex;flex-direction:column;justify-content:center}.bubble-wrapper.user-msg{align-items:end}.bubble-wrapper.ai-msg{align-items:start}.bubble-loading{display:flex;align-items:center;justify-content:flex-start;height:24px;margin-top:4px;margin-bottom:16px}.bubble-loading-dot{width:6px;height:6px;border-radius:50%;margin-right:6px;background-color:#bbb;animation:.9s linear infinite loading-dot}.bubble-loading-dot:first-child{animation-delay:0s}.bubble-loading-dot:nth-child(2){animation-delay:.1s}.bubble-loading-dot:nth-child(3){animation-delay:.2s}@keyframes loading-dot{0%,to{opacity:.3}33%,66%{opacity:1}}.message-bubble{width:100%;display:flex;align-items:flex-end;flex-direction:column;gap:8px}.message-bubble .message-content.message-human{padding:8px 16px;margin-left:auto;border-radius:12px;background:#E4EDFD;color:#004ad3}.message-bubble .message-content.message-assistant{padding:0;width:100%;background:#fff}.message-bubble .message-markdown{background:#fff}.message-bubble .message-think{width:100%;padding:12px;background:#FFFFFF;border-radius:10px;border:1px solid #D9DADB;font-size:14px}.message-bubble .message-think-header{display:flex;align-items:center;justify-content:space-between;cursor:pointer;color:#000;font-size:16px}.message-bubble .message-think-header-title{display:flex;align-items:center;gap:10px}.message-bubble .message-think-header-title img{width:14px;height:14px}.message-bubble .message-think .message-content{padding-top:12px;color:#61666b}.message-bubble .message-time{font-size:12px;color:#999}.assistant-tools{width:100%;display:flex;align-items:center;gap:24px}.assistant-tools img{width:14px;height:14px;cursor:pointer}\n"] }]
102
+ }], propDecorators: { messages: [{
103
+ type: Input
104
+ }], messagesContainer: [{
105
+ type: ViewChild,
106
+ args: ['messagesContainer']
107
+ }], regenerateAnswer: [{
108
+ type: Output
109
+ }] } });
110
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1tZXNzYWdlcy5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hZ2VudC9jaGF0L2NoYXQtbWVzc2FnZXMvY2hhdC1tZXNzYWdlcy5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hZ2VudC9jaGF0L2NoYXQtbWVzc2FnZXMvY2hhdC1tZXNzYWdlcy5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSw2QkFBNkI7QUFDN0IsT0FBTyxFQUNMLFNBQVMsRUFDVCxLQUFLLEVBQ0wsU0FBUyxFQUdULE1BQU0sRUFFTixNQUFNLEVBQ04sWUFBWSxHQUdiLE1BQU0sZUFBZSxDQUFDO0FBRXZCLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLHFDQUFxQyxDQUFDO0FBQzVFLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUVyRCxPQUFPLEVBQUUsWUFBWSxFQUFZLE1BQU0sMkJBQTJCLENBQUM7QUFDbkUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sdUJBQXVCLENBQUM7Ozs7O0FBT3pELE1BQU0sT0FBTyxxQkFBcUI7SUFMbEM7UUFNVyxhQUFRLEdBQWMsRUFBRSxDQUFDO1FBRWxDOztXQUVHO1FBQ08scUJBQWdCLEdBQUcsSUFBSSxZQUFZLEVBQVcsQ0FBQztRQUVqRCxPQUFFLEdBQUcsTUFBTSxDQUFDLHFCQUFxQixDQUFDLENBQUM7UUFDbkMsZUFBVSxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUNoQyxjQUFTLEdBQUcsTUFBTSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBQ2pDLFFBQUcsR0FBRyxNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztRQUV2QyxjQUFTLEdBQUcsS0FBSyxDQUFDO1FBQ2xCLGdCQUFXLEdBQUcsR0FBRyxDQUFDO1FBQ2xCLGlCQUFZLEdBQWEsRUFBRSxDQUFDO1FBQ3BCLHdC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S0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFDLEVBQUUsVUFBVSxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQzNELE1BQU0sYUFBYSxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXpELE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxFQUFFLENBQUMsaUJBQWlCLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDckQsa0VBQWtFO1FBQ2xFLE9BQU8sRUFBRSxRQUFRLEVBQUUsSUFBYyxFQUFFLFNBQVMsRUFBRSxhQUFhLEVBQUUsQ0FBQztJQUNoRSxDQUFDOzttSEFoR1UscUJBQXFCO3VHQUFyQixxQkFBcUIsMlJDMUJsQyxxc0dBMEVBOzRGRGhEYSxxQkFBcUI7a0JBTGpDLFNBQVM7K0JBQ0UsbUJBQW1COzhCQUtwQixRQUFRO3NCQUFoQixLQUFLO2dCQUMwQixpQkFBaUI7c0JBQWhELFNBQVM7dUJBQUMsbUJBQW1CO2dCQUlwQixnQkFBZ0I7c0JBQXpCLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyIvLyBjaGF0LW1lc3NhZ2VzLmNvbXBvbmVudC50c1xuaW1wb3J0IHtcbiAgQ29tcG9uZW50LFxuICBJbnB1dCxcbiAgVmlld0NoaWxkLFxuICBFbGVtZW50UmVmLFxuICBBZnRlclZpZXdDaGVja2VkLFxuICBpbmplY3QsXG4gIEFmdGVyVmlld0luaXQsXG4gIE91dHB1dCxcbiAgRXZlbnRFbWl0dGVyLFxuICBPbkNoYW5nZXMsXG4gIFNpbXBsZUNoYW5nZXMsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTWVzc2FnZSB9IGZyb20gJy4uL21vZGVscy9tZXNzYWdlLm1vZGVsJztcbmltcG9ydCB7IE1hcmtkb3duU3RyZWFtU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL21hcmtkb3duLXN0cmVhbS5zZXJ2aWNlJztcbmltcG9ydCB7IFNzZVNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9zc2Uuc2VydmljZSc7XG5pbXBvcnQgeyBtYXJrZWQgfSBmcm9tICdtYXJrZWQnO1xuaW1wb3J0IHsgRG9tU2FuaXRpemVyLCBTYWZlSHRtbCB9IGZyb20gJ0Bhbmd1bGFyL3BsYXRmb3JtLWJyb3dzZXInO1xuaW1wb3J0IHsgTnpNZXNzYWdlU2VydmljZSB9IGZyb20gJ25nLXpvcnJvLWFudGQvbWVzc2FnZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ25neC1jaGF0LW1lc3NhZ2VzJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2NoYXQtbWVzc2FnZXMuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZVVybHM6IFsnLi9jaGF0LW1lc3NhZ2VzLmNvbXBvbmVudC5zY3NzJ10sXG59KVxuZXhwb3J0IGNsYXNzIENoYXRNZXNzYWdlc0NvbXBvbmVudCBpbXBsZW1lbnRzIEFmdGVyVmlld0NoZWNrZWQsIEFmdGVyVmlld0luaXQsIE9uQ2hhbmdlcyB7XG4gIEBJbnB1dCgpIG1lc3NhZ2VzOiBNZXNzYWdlW10gPSBbXTtcbiAgQFZpZXdDaGlsZCgnbWVzc2FnZXNDb250YWluZXInKSBtZXNzYWdlc0NvbnRhaW5lciE6IEVsZW1lbnRSZWY7XG4gIC8qKlxuICAgKiDph43mlrDnlJ/miJDlm57nrZRcbiAgICovXG4gIEBPdXRwdXQoKSByZWdlbmVyYXRlQW5zd2VyID0gbmV3IEV2ZW50RW1pdHRlcjxNZXNzYWdlPigpO1xuXG4gIHByaXZhdGUgbWQgPSBpbmplY3QoTWFya2Rvd25TdHJlYW1TZXJ2aWNlKTtcbiAgcHJpdmF0ZSBzc2VTZXJ2aWNlID0gaW5qZWN0KFNzZVNlcnZpY2UpO1xuICBwcml2YXRlIHNhbml0aXplciA9IGluamVjdChEb21TYW5pdGl6ZXIpO1xuICBwcml2YXRlIG1zZyA9IGluamVjdChOek1lc3NhZ2VTZXJ2aWNlKTtcblxuICBpc0xvYWRpbmcgPSBmYWxzZTtcbiAgaW5wdXRIZWlnaHQgPSAxODA7XG4gIHJlbmRlcmVkSHRtbDogU2FmZUh0bWwgPSAnJztcbiAgcHJpdmF0ZSBhY2N1bXVsYXRlZE1hcmtkb3duID0gJyc7XG5cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIHRoaXMuc3NlU2VydmljZS5sb2FkaW5nJC5zdWJzY3JpYmUoc3RhdHVzID0+IHtcbiAgICAgIHRoaXMuaXNMb2FkaW5nID0gc3RhdHVzO1xuICAgIH0pO1xuICAgIHRoaXMuc3NlU2VydmljZS5kZWx0YSQuc3Vic2NyaWJlKGRlbHRhID0+IHtcbiAgICAgIGlmICghZGVsdGEpIHJldHVybjtcblxuICAgICAgdGhpcy5hY2N1bXVsYXRlZE1hcmtkb3duICs9IGRlbHRhO1xuICAgICAgdGhpcy5yZW5kZXJlZEh0bWwgPSB0aGlzLm1kLnJlbmRlckluY3JlbWVudGFsKHRoaXMuYWNjdW11bGF0ZWRNYXJrZG93bik7XG4gICAgfSk7XG4gIH1cblxuICBuZ0FmdGVyVmlld0NoZWNrZWQoKSB7XG4gICAgLy8g5q2j5Zyo5Zue562U5LitXG4gICAgaWYgKHRoaXMuaXNMb2FkaW5nKSB7XG4gICAgICB0aGlzLnNjcm9sbFRvQm90dG9tKCk7XG4gICAgfVxuICB9XG5cbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcykge1xuICAgIGlmIChjaGFuZ2VzWydtZXNzYWdlcyddKSB7XG4gICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgdGhpcy5zY3JvbGxUb0JvdHRvbSgpO1xuICAgICAgfSwgNTApO1xuICAgIH1cbiAgfVxuXG4gIC8vIOa7muWKqOWIsOW6lemDqFxuICBzY3JvbGxUb0JvdHRvbSgpOiB2b2lkIHtcbiAgICBpZiAoIXRoaXMubWVzc2FnZXNDb250YWluZXIpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5tZXNzYWdlc0NvbnRhaW5lci5uYXRpdmVFbGVtZW50LnNjcm9sbFRvcCA9IHRoaXMubWVzc2FnZXNDb250YWluZXIubmF0aXZlRWxlbWVudC5zY3JvbGxIZWlnaHQ7XG4gIH1cblxuICBjb3B5Q29udGVudCh0ZXh0OiBzdHJpbmcpOiB2b2lkIHtcbiAgICBpZiAobmF2aWdhdG9yLmNsaXBib2FyZCAmJiB3aW5kb3cuaXNTZWN1cmVDb250ZXh0KSB7XG4gICAgICAvLyDkvb/nlKjnjrDku6MgQ2xpcGJvYXJkIEFQSVxuICAgICAgbmF2aWdhdG9yLmNsaXBib2FyZC53cml0ZVRleHQodGV4dCkudGhlbigoKSA9PiB7XG4gICAgICAgIHRoaXMubXNnLnN1Y2Nlc3MoJ+WkjeWItuaIkOWKnycpO1xuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgb25QYWdlQ2hhbmdlKG51bTogbnVtYmVyLCBtc2c6IE1lc3NhZ2UpOiB2b2lkIHtcbiAgICBtc2cuYW5zd2VySW5kZXggPSBudW0gLSAxO1xuICB9XG5cbiAgLyoqXG4gICAqIOmHjeaWsOeUn+aIkOWkmuS4quWbnuetlFxuICAgKi9cbiAgcmVnZW5lcmF0ZShtc2c6IE1lc3NhZ2UpOiB2b2lkIHtcbiAgICAvLyBjb25zdCBxdWVzdGlvbiA9IHRoaXMubWVzc2FnZXNbaW5kZXggLSAxXS5tZXNzYWdlO1xuICAgIC8vIOmHjeaWsOeUn+aIkOS9v+eUqOW6lemDqOacgOaWsOeahOmFjee9ru+8jOmHjeaWsOiwg+eUqHNlbmRNZXNzYWdlXG4gICAgdGhpcy5yZWdlbmVyYXRlQW5zd2VyLmVtaXQobXNnKTtcbiAgfVxuXG4gIC8vIOaKiuWujOaVtOeahCBtYXJrZG93biDlnZfmuLLmn5PvvIzliankuIvnmoTnlZnnu5nmiZPlrZfljLpcbiAgcHJpdmF0ZSByZW5kZXJDb21wbGV0ZUJsb2NrcyhtYXJrZG93bjogc3RyaW5nKTogeyByZW5kZXJlZDogc3RyaW5nOyByZW1haW5pbmc6IHN0cmluZyB9IHtcbiAgICAvLyDmib7liLDmnIDlkI7kuIDkuKrmjaLooYznrKbkuYvliY3nmoTpg73muLLmn5PvvIjotrPlpJ/kuJ3mu5HvvIlcbiAgICBjb25zdCBsaW5lcyA9IG1hcmtkb3duLnNwbGl0KCdcXG4nKTtcbiAgICBsZXQgYnJlYWtQb2ludCA9IGxpbmVzLmxlbmd0aDtcblxuICAgIC8vIOS7juWQjuW+gOWJjeaJvu+8jOS/neeVmeacquWujOaIkOeahOS7o+eggeWdly/ooajmoLwv5YiX6KGoXG4gICAgZm9yIChsZXQgaSA9IGxpbmVzLmxlbmd0aCAtIDE7IGkgPj0gMDsgaS0tKSB7XG4gICAgICBjb25zdCBsaW5lID0gbGluZXNbaV07XG4gICAgICBpZiAobGluZS50cmltKCkgPT09ICcnIHx8IGxpbmUuc3RhcnRzV2l0aCgnYGBgJykgfHwgbGluZS5zdGFydHNXaXRoKCd8JykgfHwgbGluZS5zdGFydHNXaXRoKCctICcpKSB7XG4gICAgICAgIGJyZWFrUG9pbnQgPSBpO1xuICAgICAgICBicmVhaztcbiAgICAgIH1cbiAgICB9XG5cbiAgICBjb25zdCBjb21wbGV0ZVBhcnQgPSBsaW5lcy5zbGljZSgwLCBicmVha1BvaW50KS5qb2luKCdcXG4nKTtcbiAgICBjb25zdCByZW1haW5pbmdQYXJ0ID0gbGluZXMuc2xpY2UoYnJlYWtQb2ludCkuam9pbignXFxuJyk7XG5cbiAgICBjb25zdCBodG1sID0gdGhpcy5tZC5yZW5kZXJJbmNyZW1lbnRhbChjb21wbGV0ZVBhcnQpO1xuICAgIC8vIGNvbnN0IGh0bWwgPSBtYXJrZWQoY29tcGxldGVQYXJ0LCB7IGJyZWFrczogdHJ1ZSwgZ2ZtOiB0cnVlIH0pO1xuICAgIHJldHVybiB7IHJlbmRlcmVkOiBodG1sIGFzIHN0cmluZywgcmVtYWluaW5nOiByZW1haW5pbmdQYXJ0IH07XG4gIH1cbn1cbiIsIjwhLS0gY2hhdC1tZXNzYWdlcy5jb21wb25lbnQuaHRtbCAtLT5cbjxzY3JpcHQgc3JjPVwiY2hhdC1tZXNzYWdlcy5jb21wb25lbnQudHNcIj48L3NjcmlwdD5cbjxkaXYgY2xhc3M9XCJtZXNzYWdlcy1jb250YWluZXJcIiAjbWVzc2FnZXNDb250YWluZXI+XG4gIDxkaXYgY2xhc3M9XCJtZXNzYWdlLWxpc3RcIiBbc3R5bGUuLS1pbnB1dC1oZWlnaHQucHhdPVwiaW5wdXRIZWlnaHRcIj5cbiAgICA8ZGl2IGNsYXNzPVwicXVlc3Rpb24gbWVzc2FnZS13cmFwcGVyXCIgKm5nRm9yPVwibGV0IG1zZyBvZiBtZXNzYWdlczsgbGV0IGkgPSBpbmRleFwiPlxuICAgICAgPCEtLSBRIC0tPlxuICAgICAgPGRpdiBjbGFzcz1cImJ1YmJsZS13cmFwcGVyIHVzZXItbXNnXCI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJtZXNzYWdlLWJ1YmJsZVwiPlxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJtZXNzYWdlLWNvbnRlbnQgbWVzc2FnZS1odW1hblwiPlxuICAgICAgICAgICAge3sgbXNnLnF1ZXN0aW9uIH19XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgICA8IS0tIEEgLS0+XG4gICAgICA8ZGl2IGNsYXNzPVwiYnViYmxlLXdyYXBwZXIgYWktbXNnXCIgKm5nSWY9XCJtc2cuYW5zd2Vyc1ttc2cuYW5zd2VySW5kZXhdXCI+XG4gICAgICAgIDxkaXZcbiAgICAgICAgICBjbGFzcz1cImJ1YmJsZS1sb2FkaW5nXCJcbiAgICAgICAgICAqbmdJZj1cIiFtc2cuYW5zd2Vyc1ttc2cuYW5zd2VySW5kZXhdLnRoaW5rICYmICFtc2cuYW5zd2Vyc1ttc2cuYW5zd2VySW5kZXhdLmNvbnRlbnRcIlxuICAgICAgICA+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cImJ1YmJsZS1sb2FkaW5nLWRvdFwiICpuZ0Zvcj1cImxldCBpdGVtIG9mIFsxLCAyLCAzXVwiPjwvZGl2PlxuICAgICAgICA8L2Rpdj5cbiAgICAgICAgPGRpdiBjbGFzcz1cIm1lc3NhZ2UtYnViYmxlXCI+XG4gICAgICAgICAgPCEtLSDmgJ3ogIMgLS0+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cIm1lc3NhZ2UtdGhpbmtcIiAqbmdJZj1cIm1zZy5hbnN3ZXJzW21zZy5hbnN3ZXJJbmRleF0udGhpbmtcIj5cbiAgICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgICAgY2xhc3M9XCJtZXNzYWdlLXRoaW5rLWhlYWRlclwiXG4gICAgICAgICAgICAgIChjbGljayk9XCJtc2cuYW5zd2Vyc1ttc2cuYW5zd2VySW5kZXhdLnRoaW5rRXhwYW5kID0gIW1zZy5hbnN3ZXJzW21zZy5hbnN3ZXJJbmRleF0udGhpbmtFeHBhbmRcIlxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICA8ZGl2IGNsYXNzPVwibWVzc2FnZS10aGluay1oZWFkZXItdGl0bGVcIj5cbiAgICAgICAgICAgICAgICA8aW1nIHNyYz1cIi9hc3NldHMvaW1hZ2VzL2NhcmQvdGhpbmstZmluaXNoLnBuZ1wiIC8+XG4gICAgICAgICAgICAgICAge3sgbXNnLmFuc3dlcnNbbXNnLmFuc3dlckluZGV4XS5jb250ZW50ID8gJ+W3suWujOaIkOaAneiAgycgOiAn5q2j5Zyo5oCd6ICD5LitLi4uJyB9fVxuICAgICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICAgICAgPHN2ZyB3aWR0aD1cIjE4XCIgaGVpZ2h0PVwiMThcIiB2aWV3Qm94PVwiMCAwIDI0IDI0XCIgZmlsbD1cIm5vbmVcIiBzdHJva2U9XCJjdXJyZW50Q29sb3JcIiBzdHJva2Utd2lkdGg9XCIyXCI+XG4gICAgICAgICAgICAgICAgPHBvbHlsaW5lIHBvaW50cz1cIjYgOSAxMiAxNSAxOCA5XCI+PC9wb2x5bGluZT5cbiAgICAgICAgICAgICAgPC9zdmc+XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgICAgY2xhc3M9XCJtZXNzYWdlLWNvbnRlbnQgbWVzc2FnZS1hc3Npc3RhbnQgbWVzc2FnZS1tYXJrZG93blwiXG4gICAgICAgICAgICAgIFtpbm5lckhUTUxdPVwibXNnLmFuc3dlcnNbbXNnLmFuc3dlckluZGV4XS50aGluayB8IG1hcmtkb3duXCJcbiAgICAgICAgICAgICAgW2hpZGRlbl09XCIhbXNnLmFuc3dlcnNbbXNnLmFuc3dlckluZGV4XS50aGlua0V4cGFuZFwiXG4gICAgICAgICAgICA+PC9kaXY+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPCEtLSDlm57nrZQgLS0+XG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgY2xhc3M9XCJtZXNzYWdlLWNvbnRlbnQgbWVzc2FnZS1hc3Npc3RhbnQgbWVzc2FnZS1tYXJrZG93blwiXG4gICAgICAgICAgICBbaW5uZXJIVE1MXT1cIm1zZy5hbnN3ZXJzW21zZy5hbnN3ZXJJbmRleF0uY29udGVudCB8IG1hcmtkb3duXCJcbiAgICAgICAgICA+PC9kaXY+XG4gICAgICAgICAgPCEtLSDlm57nrZTlt6XlhbfmoI8gLS0+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cImFzc2lzdGFudC10b29sc1wiICpuZ0lmPVwibXNnLmFuc3dlcnNbbXNnLmFuc3dlckluZGV4XS5jb250ZW50XCI+XG4gICAgICAgICAgICA8bmd4LXNpbXBsZS1wYWdpbmF0aW9uXG4gICAgICAgICAgICAgIFtjdXJyZW50XT1cIm1zZy5hbnN3ZXJJbmRleCArIDFcIlxuICAgICAgICAgICAgICBbdG90YWxdPVwibXNnLmFuc3dlcnMubGVuZ3RoXCJcbiAgICAgICAgICAgICAgKHBhZ2VDaGFuZ2UpPVwib25QYWdlQ2hhbmdlKCRldmVudCwgbXNnKVwiXG4gICAgICAgICAgICAgICpuZ0lmPVwibXNnLmFuc3dlcnMubGVuZ3RoID4gMVwiXG4gICAgICAgICAgICA+PC9uZ3gtc2ltcGxlLXBhZ2luYXRpb24+XG4gICAgICAgICAgICA8aW1nXG4gICAgICAgICAgICAgIHNyYz1cImFzc2V0cy9pbWFnZXMvY2FyZC9jb3B5LnBuZ1wiXG4gICAgICAgICAgICAgIGFsdD1cIuWkjeWItlwiXG4gICAgICAgICAgICAgIChjbGljayk9XCJjb3B5Q29udGVudChtc2cuYW5zd2Vyc1ttc2cuYW5zd2VySW5kZXhdLmNvbnRlbnQpXCJcbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgICA8aW1nXG4gICAgICAgICAgICAgIHNyYz1cImFzc2V0cy9pbWFnZXMvY2FyZC9yZWZyZXNoLnBuZ1wiXG4gICAgICAgICAgICAgIGFsdD1cIumHjeaWsOeUn+aIkFwiXG4gICAgICAgICAgICAgIChjbGljayk9XCJyZWdlbmVyYXRlKG1zZylcIlxuICAgICAgICAgICAgICAqbmdJZj1cImkgPT09IG1lc3NhZ2VzLmxlbmd0aCAtIDFcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L2Rpdj5cbiAgICA8L2Rpdj5cblxuICAgIDwhLS0gPGRpdiBzdHlsZT1cInBhZGRpbmctYm90dG9tOiAxNjBweFwiPjwvZGl2PiAtLT5cbiAgPC9kaXY+XG48L2Rpdj5cbiJdfQ==
@@ -0,0 +1,118 @@
1
+ // chat-sidebar.component.ts
2
+ import { Component, inject, ViewChild } from '@angular/core';
3
+ import { animate, style, transition, trigger } from '@angular/animations';
4
+ import { HttpService } from '../services/http.service';
5
+ import { StoreService } from '../store/store.service';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "@angular/common";
8
+ import * as i2 from "./history-group/history-group.component";
9
+ export class ChatSidebarComponent {
10
+ constructor() {
11
+ this.http = inject(HttpService);
12
+ this.storeService = inject(StoreService);
13
+ this.isCollapsed = false;
14
+ this.historyGroups = [];
15
+ }
16
+ ngOnInit() {
17
+ this.getHistory();
18
+ }
19
+ // 获取历史对话
20
+ getHistory(needSelectFirst = false) {
21
+ this.http.get(`${this.storeService.getBaseUrl()}/api/ChatModel/conversation`).subscribe((res) => {
22
+ this.historyGroups = this.groupByFriendlyTime(res);
23
+ // 选中第一个对话
24
+ if (needSelectFirst) {
25
+ if (this.historyGroups[0]?.items[0]) {
26
+ this.historyGroupRef.selectConversation(this.historyGroups[0].items[0]);
27
+ }
28
+ else {
29
+ this.historyGroupRef.selectConversation(null);
30
+ }
31
+ }
32
+ });
33
+ }
34
+ // 删除后刷新,若删除当前选中会话
35
+ refresh(conversationICode) {
36
+ const needSelectFirst = this.storeService.getConversationICode() === conversationICode;
37
+ this.getHistory(needSelectFirst);
38
+ }
39
+ selectConversation(conversation) {
40
+ // 取消勾选其他会话
41
+ this.historyGroups.forEach(group => {
42
+ group.items.forEach(item => {
43
+ if (!conversation || conversation.iCode !== item.iCode) {
44
+ item.isActive = false;
45
+ }
46
+ });
47
+ });
48
+ }
49
+ toggleSidebar() {
50
+ this.isCollapsed = !this.isCollapsed;
51
+ }
52
+ openNewChat() {
53
+ this.historyGroupRef.selectConversation(null);
54
+ }
55
+ groupByFriendlyTime(dataList) {
56
+ if (!dataList || dataList.length === 0) {
57
+ return [];
58
+ }
59
+ const list = [];
60
+ const now = new Date();
61
+ const today = new Date(now.getFullYear(), now.getMonth(), now.getDate());
62
+ const yesterday = new Date(today);
63
+ yesterday.setDate(yesterday.getDate() - 1);
64
+ dataList.forEach((item, index) => {
65
+ const itemTime = new Date(item.time);
66
+ const itemDate = new Date(itemTime.getFullYear(), itemTime.getMonth(), itemTime.getDate());
67
+ let dateLabel = '';
68
+ item.isActive = false;
69
+ if (itemDate.getTime() === today.getTime()) {
70
+ dateLabel = '今天';
71
+ }
72
+ else if (itemDate.getTime() === yesterday.getTime()) {
73
+ dateLabel = '昨天';
74
+ }
75
+ else {
76
+ // 其他日期显示具体的年月日
77
+ dateLabel = `${itemTime.getFullYear()}年${(itemTime.getMonth() + 1).toString().padStart(2, '0')}月${itemTime.getDate().toString().padStart(2, '0')}日`;
78
+ }
79
+ const find = list.find(item => item.dateLabel === dateLabel);
80
+ if (find) {
81
+ find.items.push(item);
82
+ }
83
+ else {
84
+ const group = { dateLabel: '', expanded: index === 0, items: [] };
85
+ group.dateLabel = dateLabel;
86
+ group.items.push(item);
87
+ list.push(group);
88
+ }
89
+ });
90
+ return list;
91
+ }
92
+ }
93
+ ChatSidebarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ChatSidebarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
94
+ ChatSidebarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: ChatSidebarComponent, selector: "ngx-chat-sidebar", viewQueries: [{ propertyName: "historyGroupRef", first: true, predicate: ["historyGroupRef"], descendants: true }], ngImport: i0, template: "<div class=\"sidebar\" [class.collapsed]=\"isCollapsed\">\n <div class=\"sidebar-inner\">\n <div class=\"sidebar-header\">\n <div class=\"logo-area\">\n <img src=\"assets/images/logo.png\" alt=\"Logo\" class=\"logo-img\" />\n <span class=\"app-name\">\u9752\u6E05\u6C34\u5229</span>\n </div>\n <button class=\"toggle-btn\" (click)=\"toggleSidebar()\">\n <img src=\"assets/images/sidebar/collapse.png\" />\n </button>\n </div>\n\n <div class=\"sidebar-content\">\n <button class=\"new-chat-btn\" (click)=\"openNewChat()\">\n <img src=\"assets/images/sidebar/new-chat.png\" />\n \u5F00\u542F\u65B0\u5BF9\u8BDD\n </button>\n\n <div class=\"divider\"></div>\n\n <div class=\"section-title\">\n <img src=\"assets/images/sidebar/history-chat.png\" />\n <span>\u5386\u53F2\u5BF9\u8BDD</span>\n </div>\n\n <div class=\"history-list-scroll\" *ngIf=\"historyGroups.length\">\n <history-group\n #historyGroupRef\n *ngFor=\"let group of historyGroups\"\n [group]=\"group\"\n (select)=\"selectConversation($event)\"\n (refresh)=\"refresh($event)\"\n >\n </history-group>\n </div>\n\n <div class=\"history-empty\" *ngIf=\"!historyGroups.length\">\u6682\u65E0\u5386\u53F2\u5BF9\u8BDD</div>\n </div>\n </div>\n</div>\n\n<div class=\"logo-actions\" [@logoActionsAnimation]=\"isCollapsed ? 'leave' : 'enter'\" *ngIf=\"isCollapsed\">\n <img src=\"assets/images/logo.png\" alt=\"Logo\" class=\"logo-img\" />\n <div class=\"left-top-bar\">\n <img src=\"assets/images/sidebar/expand.png\" (click)=\"toggleSidebar()\" />\n <img src=\"assets/images/sidebar/new-chat.png\" (click)=\"openNewChat()\" />\n </div>\n</div>\n", styles: ["@charset \"UTF-8\";:host{display:block;height:100vh}.sidebar{width:280px;height:100%;background-color:#f3f4f6;border-right:1px solid #e0e0e0;display:flex;flex-direction:column;transition:width .4s cubic-bezier(.25,.8,.25,1),border .4s;overflow:hidden;box-sizing:border-box}.sidebar.collapsed{width:0;padding:0;border-right:0 solid transparent}.sidebar-inner{width:280px;height:100%;display:flex;flex-direction:column;padding:16px;box-sizing:border-box}.sidebar-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:20px;height:32px}.logo-area{display:flex;align-items:center;gap:8px;white-space:nowrap}.logo-img{width:28px;height:28px;object-fit:contain}.app-name{font-weight:600;font-size:16px;color:#000}.toggle-btn{background:none;border:none;cursor:pointer;color:#666;padding:4px;border-radius:4px}.toggle-btn img{width:16px;height:15px}.toggle-btn:hover{background-color:#0000000d}.sidebar-content{flex:1;display:flex;flex-direction:column;overflow:hidden}.new-chat-btn{width:100%;background-color:#fff;border:1px solid #e0e0e0;border-radius:20px;padding:10px;color:#333;font-weight:500;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:8px;box-shadow:0 2px 5px #00000008;transition:all .2s;white-space:nowrap}.new-chat-btn img{width:16px;height:16px}.new-chat-btn:hover{box-shadow:0 4px 8px #00000014;border-color:#d0d0d0}.divider{height:1px;background-color:#e0e0e0;margin:20px 0}.section-title{display:flex;align-items:center;gap:8px;font-weight:600;color:#333;margin-bottom:10px;white-space:nowrap}.section-title img{width:16px;height:16px}.history-list-scroll{flex:1;overflow-y:auto;overflow-x:hidden;padding-right:4px}.history-empty{flex:1;display:flex;align-items:center;justify-content:center}.logo-actions{position:absolute;left:12px;top:12px;display:flex;align-items:center;gap:8px;z-index:99}.logo-actions .logo-img{width:40px;height:32px}.logo-actions .left-top-bar{width:75px;height:40px;display:flex;align-items:center;justify-content:center;gap:19px;background:#FFFFFF;box-shadow:0 2px 4px #0000001a;border-radius:20px;border:1px solid #EEEEEE}.logo-actions .left-top-bar img{width:16px;height:16px;cursor:pointer}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.HistoryGroupComponent, selector: "history-group", inputs: ["group"], outputs: ["select", "refresh"] }], animations: [
95
+ trigger('logoActionsAnimation', [
96
+ transition(':enter', [
97
+ style({ opacity: 0, transform: 'translateX(20px)' }),
98
+ animate('600ms ease-out', style({ opacity: 1, transform: 'translateX(0)' })),
99
+ ]),
100
+ transition(':leave', [animate('0ms ease-in', style({ opacity: 0, transform: 'translateX(20px)' }))]),
101
+ ]),
102
+ ] });
103
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: ChatSidebarComponent, decorators: [{
104
+ type: Component,
105
+ args: [{ selector: 'ngx-chat-sidebar', animations: [
106
+ trigger('logoActionsAnimation', [
107
+ transition(':enter', [
108
+ style({ opacity: 0, transform: 'translateX(20px)' }),
109
+ animate('600ms ease-out', style({ opacity: 1, transform: 'translateX(0)' })),
110
+ ]),
111
+ transition(':leave', [animate('0ms ease-in', style({ opacity: 0, transform: 'translateX(20px)' }))]),
112
+ ]),
113
+ ], template: "<div class=\"sidebar\" [class.collapsed]=\"isCollapsed\">\n <div class=\"sidebar-inner\">\n <div class=\"sidebar-header\">\n <div class=\"logo-area\">\n <img src=\"assets/images/logo.png\" alt=\"Logo\" class=\"logo-img\" />\n <span class=\"app-name\">\u9752\u6E05\u6C34\u5229</span>\n </div>\n <button class=\"toggle-btn\" (click)=\"toggleSidebar()\">\n <img src=\"assets/images/sidebar/collapse.png\" />\n </button>\n </div>\n\n <div class=\"sidebar-content\">\n <button class=\"new-chat-btn\" (click)=\"openNewChat()\">\n <img src=\"assets/images/sidebar/new-chat.png\" />\n \u5F00\u542F\u65B0\u5BF9\u8BDD\n </button>\n\n <div class=\"divider\"></div>\n\n <div class=\"section-title\">\n <img src=\"assets/images/sidebar/history-chat.png\" />\n <span>\u5386\u53F2\u5BF9\u8BDD</span>\n </div>\n\n <div class=\"history-list-scroll\" *ngIf=\"historyGroups.length\">\n <history-group\n #historyGroupRef\n *ngFor=\"let group of historyGroups\"\n [group]=\"group\"\n (select)=\"selectConversation($event)\"\n (refresh)=\"refresh($event)\"\n >\n </history-group>\n </div>\n\n <div class=\"history-empty\" *ngIf=\"!historyGroups.length\">\u6682\u65E0\u5386\u53F2\u5BF9\u8BDD</div>\n </div>\n </div>\n</div>\n\n<div class=\"logo-actions\" [@logoActionsAnimation]=\"isCollapsed ? 'leave' : 'enter'\" *ngIf=\"isCollapsed\">\n <img src=\"assets/images/logo.png\" alt=\"Logo\" class=\"logo-img\" />\n <div class=\"left-top-bar\">\n <img src=\"assets/images/sidebar/expand.png\" (click)=\"toggleSidebar()\" />\n <img src=\"assets/images/sidebar/new-chat.png\" (click)=\"openNewChat()\" />\n </div>\n</div>\n", styles: ["@charset \"UTF-8\";:host{display:block;height:100vh}.sidebar{width:280px;height:100%;background-color:#f3f4f6;border-right:1px solid #e0e0e0;display:flex;flex-direction:column;transition:width .4s cubic-bezier(.25,.8,.25,1),border .4s;overflow:hidden;box-sizing:border-box}.sidebar.collapsed{width:0;padding:0;border-right:0 solid transparent}.sidebar-inner{width:280px;height:100%;display:flex;flex-direction:column;padding:16px;box-sizing:border-box}.sidebar-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:20px;height:32px}.logo-area{display:flex;align-items:center;gap:8px;white-space:nowrap}.logo-img{width:28px;height:28px;object-fit:contain}.app-name{font-weight:600;font-size:16px;color:#000}.toggle-btn{background:none;border:none;cursor:pointer;color:#666;padding:4px;border-radius:4px}.toggle-btn img{width:16px;height:15px}.toggle-btn:hover{background-color:#0000000d}.sidebar-content{flex:1;display:flex;flex-direction:column;overflow:hidden}.new-chat-btn{width:100%;background-color:#fff;border:1px solid #e0e0e0;border-radius:20px;padding:10px;color:#333;font-weight:500;cursor:pointer;display:flex;align-items:center;justify-content:center;gap:8px;box-shadow:0 2px 5px #00000008;transition:all .2s;white-space:nowrap}.new-chat-btn img{width:16px;height:16px}.new-chat-btn:hover{box-shadow:0 4px 8px #00000014;border-color:#d0d0d0}.divider{height:1px;background-color:#e0e0e0;margin:20px 0}.section-title{display:flex;align-items:center;gap:8px;font-weight:600;color:#333;margin-bottom:10px;white-space:nowrap}.section-title img{width:16px;height:16px}.history-list-scroll{flex:1;overflow-y:auto;overflow-x:hidden;padding-right:4px}.history-empty{flex:1;display:flex;align-items:center;justify-content:center}.logo-actions{position:absolute;left:12px;top:12px;display:flex;align-items:center;gap:8px;z-index:99}.logo-actions .logo-img{width:40px;height:32px}.logo-actions .left-top-bar{width:75px;height:40px;display:flex;align-items:center;justify-content:center;gap:19px;background:#FFFFFF;box-shadow:0 2px 4px #0000001a;border-radius:20px;border:1px solid #EEEEEE}.logo-actions .left-top-bar img{width:16px;height:16px;cursor:pointer}\n"] }]
114
+ }], propDecorators: { historyGroupRef: [{
115
+ type: ViewChild,
116
+ args: ['historyGroupRef']
117
+ }] } });
118
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1zaWRlYmFyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2FnZW50L2NoYXQvY2hhdC1zaWRlYmFyL2NoYXQtc2lkZWJhci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9hZ2VudC9jaGF0L2NoYXQtc2lkZWJhci9jaGF0LXNpZGViYXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsNEJBQTRCO0FBQzVCLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFVLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUVyRSxPQUFPLEVBQUUsT0FBTyxFQUFFLEtBQUssRUFBRSxVQUFVLEVBQUUsT0FBTyxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDMUUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ3ZELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7OztBQWlCdEQsTUFBTSxPQUFPLG9CQUFvQjtJQWRqQztRQWdCVSxTQUFJLEdBQUcsTUFBTSxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQzNCLGlCQUFZLEdBQUcsTUFBTSxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBRTVDLGdCQUFXLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLGtCQUFhLEdBQXdCLEVBQUUsQ0FBQztLQXFGekM7SUFuRkMsUUFBUTtRQUNOLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztJQUNwQixDQUFDO0lBRUQsU0FBUztJQUNULFVBQVUsQ0FBQyxlQUFlLEdBQUcsS0FBSztRQUNoQyxJQUFJLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxHQUFHLElBQUksQ0FBQyxZQUFZLENBQUMsVUFBVSxFQUFFLDZCQUE2QixDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsR0FBbUIsRUFBRSxFQUFFO1lBQzlHLElBQUksQ0FBQyxhQUFhLEdBQUcs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QyxFQUFFLFNBQVMsRUFBRSxrQkFBa0IsRUFBRSxDQUFDO2dCQUNwRCxPQUFPLENBQUMsZ0JBQWdCLEVBQUUsS0FBSyxDQUFDLEVBQUUsT0FBTyxFQUFFLENBQUMsRUFBRSxTQUFTLEVBQUUsZUFBZSxFQUFFLENBQUMsQ0FBQzthQUM3RSxDQUFDO1lBQ0YsVUFBVSxDQUFDLFFBQVEsRUFBRSxDQUFDLE9BQU8sQ0FBQyxhQUFhLEVBQUUsS0FBSyxDQUFDLEVBQUUsT0FBTyxFQUFFLENBQUMsRUFBRSxTQUFTLEVBQUUsa0JBQWtCLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQztTQUNyRyxDQUFDO0tBQ0g7NEZBRVUsb0JBQW9CO2tCQWRoQyxTQUFTOytCQUNFLGtCQUFrQixjQUdoQjt3QkFDVixPQUFPLENBQUMsc0JBQXNCLEVBQUU7NEJBQzlCLFVBQVUsQ0FBQyxRQUFRLEVBQUU7Z0NBQ25CLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLEVBQUUsU0FBUyxFQUFFLGtCQUFrQixFQUFFLENBQUM7Z0NBQ3BELE9BQU8sQ0FBQyxnQkFBZ0IsRUFBRSxLQUFLLENBQUMsRUFBRSxPQUFPLEVBQUUsQ0FBQyxFQUFFLFNBQVMsRUFBRSxlQUFlLEVBQUUsQ0FBQyxDQUFDOzZCQUM3RSxDQUFDOzRCQUNGLFVBQVUsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxPQUFPLENBQUMsYUFBYSxFQUFFLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLEVBQUUsU0FBUyxFQUFFLGtCQUFrQixFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7eUJBQ3JHLENBQUM7cUJBQ0g7OEJBRzZCLGVBQWU7c0JBQTVDLFNBQVM7dUJBQUMsaUJBQWlCIiwic291cmNlc0NvbnRlbnQiOlsiLy8gY2hhdC1zaWRlYmFyLmNvbXBvbmVudC50c1xuaW1wb3J0IHsgQ29tcG9uZW50LCBpbmplY3QsIE9uSW5pdCwgVmlld0NoaWxkIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb252ZXJzYXRpb24sIENvbnZlcnNhdGlvbkdyb3VwIH0gZnJvbSAnLi4vbW9kZWxzL2NvbnZlcnNhdGlvbi5tb2RlbCc7XG5pbXBvcnQgeyBhbmltYXRlLCBzdHlsZSwgdHJhbnNpdGlvbiwgdHJpZ2dlciB9IGZyb20gJ0Bhbmd1bGFyL2FuaW1hdGlvbnMnO1xuaW1wb3J0IHsgSHR0cFNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9odHRwLnNlcnZpY2UnO1xuaW1wb3J0IHsgU3RvcmVTZXJ2aWNlIH0gZnJvbSAnLi4vc3RvcmUvc3RvcmUuc2VydmljZSc7XG5pbXBvcnQgeyBIaXN0b3J5R3JvdXBDb21wb25lbnQgfSBmcm9tICcuL2hpc3RvcnktZ3JvdXAvaGlzdG9yeS1ncm91cC5jb21wb25lbnQnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICduZ3gtY2hhdC1zaWRlYmFyJyxcbiAgdGVtcGxhdGVVcmw6ICcuL2NoYXQtc2lkZWJhci5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL2NoYXQtc2lkZWJhci5jb21wb25lbnQuc2NzcyddLFxuICBhbmltYXRpb25zOiBbXG4gICAgdHJpZ2dlcignbG9nb0FjdGlvbnNBbmltYXRpb24nLCBbXG4gICAgICB0cmFuc2l0aW9uKCc6ZW50ZXInLCBbXG4gICAgICAgIHN0eWxlKHsgb3BhY2l0eTogMCwgdHJhbnNmb3JtOiAndHJhbnNsYXRlWCgyMHB4KScgfSksXG4gICAgICAgIGFuaW1hdGUoJzYwMG1zIGVhc2Utb3V0Jywgc3R5bGUoeyBvcGFjaXR5OiAxLCB0cmFuc2Zvcm06ICd0cmFuc2xhdGVYKDApJyB9KSksXG4gICAgICBdKSxcbiAgICAgIHRyYW5zaXRpb24oJzpsZWF2ZScsIFthbmltYXRlKCcwbXMgZWFzZS1pbicsIHN0eWxlKHsgb3BhY2l0eTogMCwgdHJhbnNmb3JtOiAndHJhbnNsYXRlWCgyMHB4KScgfSkpXSksXG4gICAgXSksXG4gIF0sXG59KVxuZXhwb3J0IGNsYXNzIENoYXRTaWRlYmFyQ29tcG9uZW50IGltcGxlbWVudHMgT25Jbml0IHtcbiAgQFZpZXdDaGlsZCgnaGlzdG9yeUdyb3VwUmVmJykgaGlzdG9yeUdyb3VwUmVmITogSGlzdG9yeUdyb3VwQ29tcG9uZW50O1xuICBwcml2YXRlIGh0dHAgPSBpbmplY3QoSHR0cFNlcnZpY2UpO1xuICBwcml2YXRlIHN0b3JlU2VydmljZSA9IGluamVjdChTdG9yZVNlcnZpY2UpO1xuXG4gIGlzQ29sbGFwc2VkID0gZmFsc2U7XG4gIGhpc3RvcnlHcm91cHM6IENvbnZlcnNhdGlvbkdyb3VwW10gPSBbXTtcblxuICBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLmdldEhpc3RvcnkoKTtcbiAgfVxuXG4gIC8vIOiOt+WPluWOhuWPsuWvueivnVxuICBnZXRIaXN0b3J5KG5lZWRTZWxlY3RGaXJzdCA9IGZhbHNlKTogdm9pZCB7XG4gICAgdGhpcy5odHRwLmdldChgJHt0aGlzLnN0b3JlU2VydmljZS5nZXRCYXNlVXJsKCl9L2FwaS9DaGF0TW9kZWwvY29udmVyc2F0aW9uYCkuc3Vic2NyaWJlKChyZXM6IENvbnZlcnNhdGlvbltdKSA9PiB7XG4gICAgICB0aGlzLmhpc3RvcnlHcm91cHMgPSB0aGlzLmdyb3VwQnlGcmllbmRseVRpbWUocmVzKTtcbiAgICAgIC8vIOmAieS4reesrOS4gOS4quWvueivnVxuICAgICAgaWYgKG5lZWRTZWxlY3RGaXJzdCkge1xuICAgICAgICBpZiAodGhpcy5oaXN0b3J5R3JvdXBzWzBdPy5pdGVtc1swXSkge1xuICAgICAgICAgIHRoaXMuaGlzdG9yeUdyb3VwUmVmLnNlbGVjdENvbnZlcnNhdGlvbih0aGlzLmhpc3RvcnlHcm91cHNbMF0uaXRlbXNbMF0pO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHRoaXMuaGlzdG9yeUdyb3VwUmVmLnNlbGVjdENvbnZlcnNhdGlvbihudWxsKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0pO1xuICB9XG5cbiAgLy8g5Yig6Zmk5ZCO5Yi35paw77yM6Iul5Yig6Zmk5b2T5YmN6YCJ5Lit5Lya6K+dXG4gIHJlZnJlc2goY29udmVyc2F0aW9uSUNvZGU6IHN0cmluZyk6IHZvaWQge1xuICAgIGNvbnN0IG5lZWRTZWxlY3RGaXJzdCA9IHRoaXMuc3RvcmVTZXJ2aWNlLmdldENvbnZlcnNhdGlvbklDb2RlKCkgPT09IGNvbnZlcnNhdGlvbklDb2RlO1xuICAgIHRoaXMuZ2V0SGlzdG9yeShuZWVkU2VsZWN0Rmlyc3QpO1xuICB9XG5cbiAgc2VsZWN0Q29udmVyc2F0aW9uKGNvbnZlcnNhdGlvbjogQ29udmVyc2F0aW9uIHwgbnVsbCk6IHZvaWQge1xuICAgIC8vIOWPlua2iOWLvumAieWFtuS7luS8muivnVxuICAgIHRoaXMuaGlzdG9yeUdyb3Vwcy5mb3JFYWNoKGdyb3VwID0+IHtcbiAgICAgIGdyb3VwLml0ZW1zLmZvckVhY2goaXRlbSA9PiB7XG4gICAgICAgIGlmICghY29udmVyc2F0aW9uIHx8IGNvbnZlcnNhdGlvbi5pQ29kZSAhPT0gaXRlbS5pQ29kZSkge1xuICAgICAgICAgIGl0ZW0uaXNBY3RpdmUgPSBmYWxzZTtcbiAgICAgICAgfVxuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICB0b2dnbGVTaWRlYmFyKCkge1xuICAgIHRoaXMuaXNDb2xsYXBzZWQgPSAhdGhpcy5pc0NvbGxhcHNlZDtcbiAgfVxuXG4gIG9wZW5OZXdDaGF0KCk6IHZvaWQge1xuICAgIHRoaXMuaGlzdG9yeUdyb3VwUmVmLnNlbGVjdENvbnZlcnNhdGlvbihudWxsKTtcbiAgfVxuXG4gIGdyb3VwQnlGcmllbmRseVRpbWUoZGF0YUxpc3Q6IENvbnZlcnNhdGlvbltdKTogQ29udmVyc2F0aW9uR3JvdXBbXSB7XG4gICAgaWYgKCFkYXRhTGlzdCB8fCBkYXRhTGlzdC5sZW5ndGggPT09IDApIHtcbiAgICAgIHJldHVybiBbXTtcbiAgICB9XG5cbiAgICBjb25zdCBsaXN0OiBDb252ZXJzYXRpb25Hcm91cFtdID0gW107XG4gICAgY29uc3Qgbm93ID0gbmV3IERhdGUoKTtcbiAgICBjb25zdCB0b2RheSA9IG5ldyBEYXRlKG5vdy5nZXRGdWxsWWVhcigpLCBub3cuZ2V0TW9udGgoKSwgbm93LmdldERhdGUoKSk7XG4gICAgY29uc3QgeWVzdGVyZGF5ID0gbmV3IERhdGUodG9kYXkpO1xuICAgIHllc3RlcmRheS5zZXREYXRlKHllc3RlcmRheS5nZXREYXRlKCkgLSAxKTtcblxuICAgIGRhdGFMaXN0LmZvckVhY2goKGl0ZW0sIGluZGV4KSA9PiB7XG4gICAgICBjb25zdCBpdGVtVGltZSA9IG5ldyBEYXRlKGl0ZW0udGltZSEpO1xuICAgICAgY29uc3QgaXRlbURhdGUgPSBuZXcgRGF0ZShpdGVtVGltZS5nZXRGdWxsWWVhcigpLCBpdGVtVGltZS5nZXRNb250aCgpLCBpdGVtVGltZS5nZXREYXRlKCkpO1xuICAgICAgbGV0IGRhdGVMYWJlbCA9ICcnO1xuICAgICAgaXRlbS5pc0FjdGl2ZSA9IGZhbHNlO1xuXG4gICAgICBpZiAoaXRlbURhdGUuZ2V0VGltZSgpID09PSB0b2RheS5nZXRUaW1lKCkpIHtcbiAgICAgICAgZGF0ZUxhYmVsID0gJ+S7iuWkqSc7XG4gICAgICB9IGVsc2UgaWYgKGl0ZW1EYXRlLmdldFRpbWUoKSA9PT0geWVzdGVyZGF5LmdldFRpbWUoKSkge1xuICAgICAgICBkYXRlTGFiZWwgPSAn5pio5aSpJztcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIOWFtuS7luaXpeacn+aYvuekuuWFt+S9k+eahOW5tOaciOaXpVxuICAgICAgICBkYXRlTGFiZWwgPSBgJHtpdGVtVGltZS5nZXRGdWxsWWVhcigpfeW5tCR7KGl0ZW1UaW1lLmdldE1vbnRoKCkgKyAxKS50b1N0cmluZygpLnBhZFN0YXJ0KDIsICcwJyl95pyIJHtpdGVtVGltZS5nZXREYXRlKCkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpfeaXpWA7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGZpbmQgPSBsaXN0LmZpbmQoaXRlbSA9PiBpdGVtLmRhdGVMYWJlbCA9PT0gZGF0ZUxhYmVsKTtcbiAgICAgIGlmIChmaW5kKSB7XG4gICAgICAgIGZpbmQuaXRlbXMucHVzaChpdGVtKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IGdyb3VwOiBDb252ZXJzYXRpb25Hcm91cCA9IHsgZGF0ZUxhYmVsOiAnJywgZXhwYW5kZWQ6IGluZGV4ID09PSAwLCBpdGVtczogW10gfTtcbiAgICAgICAgZ3JvdXAuZGF0ZUxhYmVsID0gZGF0ZUxhYmVsO1xuICAgICAgICBncm91cC5pdGVtcy5wdXNoKGl0ZW0pO1xuICAgICAgICBsaXN0LnB1c2goZ3JvdXApO1xuICAgICAgfVxuICAgIH0pO1xuXG4gICAgcmV0dXJuIGxpc3Q7XG4gIH1cbn1cbiIsIjxkaXYgY2xhc3M9XCJzaWRlYmFyXCIgW2NsYXNzLmNvbGxhcHNlZF09XCJpc0NvbGxhcHNlZFwiPlxuICA8ZGl2IGNsYXNzPVwic2lkZWJhci1pbm5lclwiPlxuICAgIDxkaXYgY2xhc3M9XCJzaWRlYmFyLWhlYWRlclwiPlxuICAgICAgPGRpdiBjbGFzcz1cImxvZ28tYXJlYVwiPlxuICAgICAgICA8aW1nIHNyYz1cImFzc2V0cy9pbWFnZXMvbG9nby5wbmdcIiBhbHQ9XCJMb2dvXCIgY2xhc3M9XCJsb2dvLWltZ1wiIC8+XG4gICAgICAgIDxzcGFuIGNsYXNzPVwiYXBwLW5hbWVcIj7pnZLmuIXmsLTliKk8L3NwYW4+XG4gICAgICA8L2Rpdj5cbiAgICAgIDxidXR0b24gY2xhc3M9XCJ0b2dnbGUtYnRuXCIgKGNsaWNrKT1cInRvZ2dsZVNpZGViYXIoKVwiPlxuICAgICAgICA8aW1nIHNyYz1cImFzc2V0cy9pbWFnZXMvc2lkZWJhci9jb2xsYXBzZS5wbmdcIiAvPlxuICAgICAgPC9idXR0b24+XG4gICAgPC9kaXY+XG5cbiAgICA8ZGl2IGNsYXNzPVwic2lkZWJhci1jb250ZW50XCI+XG4gICAgICA8YnV0dG9uIGNsYXNzPVwibmV3LWNoYXQtYnRuXCIgKGNsaWNrKT1cIm9wZW5OZXdDaGF0KClcIj5cbiAgICAgICAgPGltZyBzcmM9XCJhc3NldHMvaW1hZ2VzL3NpZGViYXIvbmV3LWNoYXQucG5nXCIgLz5cbiAgICAgICAg5byA5ZCv5paw5a+56K+dXG4gICAgICA8L2J1dHRvbj5cblxuICAgICAgPGRpdiBjbGFzcz1cImRpdmlkZXJcIj48L2Rpdj5cblxuICAgICAgPGRpdiBjbGFzcz1cInNlY3Rpb24tdGl0bGVcIj5cbiAgICAgICAgPGltZyBzcmM9XCJhc3NldHMvaW1hZ2VzL3NpZGViYXIvaGlzdG9yeS1jaGF0LnBuZ1wiIC8+XG4gICAgICAgIDxzcGFuPuWOhuWPsuWvueivnTwvc3Bhbj5cbiAgICAgIDwvZGl2PlxuXG4gICAgICA8ZGl2IGNsYXNzPVwiaGlzdG9yeS1saXN0LXNjcm9sbFwiICpuZ0lmPVwiaGlzdG9yeUdyb3Vwcy5sZW5ndGhcIj5cbiAgICAgICAgPGhpc3RvcnktZ3JvdXBcbiAgICAgICAgICAjaGlzdG9yeUdyb3VwUmVmXG4gICAgICAgICAgKm5nRm9yPVwibGV0IGdyb3VwIG9mIGhpc3RvcnlHcm91cHNcIlxuICAgICAgICAgIFtncm91cF09XCJncm91cFwiXG4gICAgICAgICAgKHNlbGVjdCk9XCJzZWxlY3RDb252ZXJzYXRpb24oJGV2ZW50KVwiXG4gICAgICAgICAgKHJlZnJlc2gpPVwicmVmcmVzaCgkZXZlbnQpXCJcbiAgICAgICAgPlxuICAgICAgICA8L2hpc3RvcnktZ3JvdXA+XG4gICAgICA8L2Rpdj5cblxuICAgICAgPGRpdiBjbGFzcz1cImhpc3RvcnktZW1wdHlcIiAqbmdJZj1cIiFoaXN0b3J5R3JvdXBzLmxlbmd0aFwiPuaaguaXoOWOhuWPsuWvueivnTwvZGl2PlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cbjwvZGl2PlxuXG48ZGl2IGNsYXNzPVwibG9nby1hY3Rpb25zXCIgW0Bsb2dvQWN0aW9uc0FuaW1hdGlvbl09XCJpc0NvbGxhcHNlZCA/ICdsZWF2ZScgOiAnZW50ZXInXCIgKm5nSWY9XCJpc0NvbGxhcHNlZFwiPlxuICA8aW1nIHNyYz1cImFzc2V0cy9pbWFnZXMvbG9nby5wbmdcIiBhbHQ9XCJMb2dvXCIgY2xhc3M9XCJsb2dvLWltZ1wiIC8+XG4gIDxkaXYgY2xhc3M9XCJsZWZ0LXRvcC1iYXJcIj5cbiAgICA8aW1nIHNyYz1cImFzc2V0cy9pbWFnZXMvc2lkZWJhci9leHBhbmQucG5nXCIgKGNsaWNrKT1cInRvZ2dsZVNpZGViYXIoKVwiIC8+XG4gICAgPGltZyBzcmM9XCJhc3NldHMvaW1hZ2VzL3NpZGViYXIvbmV3LWNoYXQucG5nXCIgKGNsaWNrKT1cIm9wZW5OZXdDaGF0KClcIiAvPlxuICA8L2Rpdj5cbjwvZGl2PlxuIl19