diff --git a/bbcode/Editor.vue b/bbcode/Editor.vue index c8d6313..4e51ed5 100644 --- a/bbcode/Editor.vue +++ b/bbcode/Editor.vue @@ -102,6 +102,7 @@ this.sizer = this.$refs['sizer']; this.sizer.style.cssText = styles.cssText; this.sizer.style.height = '0'; + this.sizer.style.minHeight = '0'; this.sizer.style.overflow = 'hidden'; this.sizer.style.position = 'absolute'; this.sizer.style.top = '0'; diff --git a/bbcode/editor.ts b/bbcode/editor.ts index 2cd1757..279ce55 100644 --- a/bbcode/editor.ts +++ b/bbcode/editor.ts @@ -45,7 +45,7 @@ export let defaultButtons: ReadonlyArray = [ key: Keys.KeyS }, { - title: 'Color (Ctrl+D)\n\nStyles text with a color. Valid colors are: red, orange, yellow, green, cyan, blue, purple, pink, black, white, gray, primary, secondary, accent, and contrast.', + title: 'Color (Ctrl+D)\n\nStyles text with a color. Valid colors are: red, orange, yellow, green, cyan, blue, purple, pink, black, brown, white and gray.', tag: 'color', startText: '[color=]', icon: 'fa-eye-dropper', diff --git a/chat/ChannelList.vue b/chat/ChannelList.vue index 8458d98..d2c725d 100644 --- a/chat/ChannelList.vue +++ b/chat/ChannelList.vue @@ -1,5 +1,5 @@
-
+
+
+ +
-
+
+ {{s.option && ((s.option.key[0] == '#' ? '#' : '') + s.option.name) || l('logs.selectConversation')}} +
+
+ +
-
+
-
+
@@ -58,18 +67,19 @@ import FilterableSelect from '../components/FilterableSelect.vue'; import Modal from '../components/Modal.vue'; import {Keys} from '../keys'; - import {getKey, messageToString} from './common'; + import {formatTime, getKey, messageToString} from './common'; import core from './core'; import {Conversation, Logs as LogInterface} from './interfaces'; import l from './localize'; import MessageView from './message_view'; + import Zip from './zip'; function formatDate(this: void, date: Date): string { return format(date, 'YYYY-MM-DD'); } - function formatTime(this: void, date: Date): string { - return format(date, 'YYYY-MM-DD HH:mm'); + function getLogs(messages: ReadonlyArray): string { + return messages.reduce((acc, x) => acc + messageToString(x, (date) => formatTime(date, true)), ''); } @Component({ @@ -91,6 +101,7 @@ characters: ReadonlyArray = []; selectedCharacter = core.connection.character; showFilters = true; + canZip = core.logs.canZip; get filteredMessages(): ReadonlyArray { if(this.filter.length === 0) return this.messages; @@ -131,22 +142,47 @@ await this.loadMessages(); } - download(file: string, logs: ReadonlyArray): void { + download(file: string, logs: string): void { const a = document.createElement('a'); - a.target = '_blank'; - a.href = `data:${encodeURIComponent(file)},${encodeURIComponent(logs.map((x) => messageToString(x, formatTime)).join(''))}`; + a.href = logs; a.setAttribute('download', file); a.style.display = 'none'; document.body.appendChild(a); setTimeout(() => { a.click(); document.body.removeChild(a); + URL.revokeObjectURL(logs); }); } downloadDay(): void { if(this.selectedConversation === null || this.selectedDate === null || this.messages.length === 0) return; - this.download(`${this.selectedConversation.name}-${formatDate(new Date(this.selectedDate))}.txt`, this.messages); + const name = `${this.selectedConversation.name}-${formatDate(new Date(this.selectedDate))}.txt`; + this.download(name, `data:${encodeURIComponent(name)},${encodeURIComponent(getLogs(this.messages))}`); + } + + async downloadConversation(): Promise { + if(this.selectedConversation === null) return; + const zip = new Zip(); + for(const date of this.dates) { + const messages = await core.logs.getLogs(this.selectedCharacter, this.selectedConversation.key, date); + zip.addFile(`${formatDate(date)}.txt`, getLogs(messages)); + } + this.download(`${this.selectedConversation.name}.zip`, URL.createObjectURL(zip.build())); + } + + async downloadCharacter(): Promise { + if(this.selectedCharacter === '' || !confirm(l('logs.confirmExport', this.selectedCharacter))) return; + const zip = new Zip(); + for(const conv of this.conversations) { + zip.addFile(`${conv.name}/`, ''); + const dates = await core.logs.getLogDates(this.selectedCharacter, conv.key); + for(const date of dates) { + const messages = await core.logs.getLogs(this.selectedCharacter, conv.key, date); + zip.addFile(`${conv.name}/${formatDate(date)}.txt`, getLogs(messages)); + } + } + this.download(`${this.selectedCharacter}.zip`, URL.createObjectURL(zip.build())); } async onOpen(): Promise { diff --git a/chat/RecentConversations.vue b/chat/RecentConversations.vue index 35d1b8a..cb2c59a 100644 --- a/chat/RecentConversations.vue +++ b/chat/RecentConversations.vue @@ -13,7 +13,7 @@ import Component from 'vue-class-component'; import CustomDialog from '../components/custom_dialog'; import Modal from '../components/Modal.vue'; - import ChannelView from './ChannelView.vue'; + import ChannelView from './ChannelTagView.vue'; import core from './core'; import {Character, Conversation} from './interfaces'; import l from './localize'; diff --git a/chat/SettingsView.vue b/chat/SettingsView.vue index 94590d0..103b8fd 100644 --- a/chat/SettingsView.vue +++ b/chat/SettingsView.vue @@ -1,7 +1,8 @@