diff --git a/bbcode/Editor.vue b/bbcode/Editor.vue index 0559702..bd9e346 100644 --- a/bbcode/Editor.vue +++ b/bbcode/Editor.vue @@ -38,26 +38,25 @@ \ No newline at end of file + diff --git a/bbcode/core.ts b/bbcode/core.ts index bf7e71e..921edc7 100644 --- a/bbcode/core.ts +++ b/bbcode/core.ts @@ -4,6 +4,8 @@ const urlFormat = '((?:https?|ftps?|irc)://[^\\s/$.?#"\']+\\.[^\\s"]+)'; export const findUrlRegex = new RegExp(`(\\[url[=\\]]\\s*)?${urlFormat}`, 'gi'); export const urlRegex = new RegExp(`^${urlFormat}$`); +export type BBCodeElement = HTMLElement & {cleanup?(): void}; + export function domain(url: string): string | undefined { const pieces = urlRegex.exec(url); if(pieces === null) return; @@ -111,4 +113,4 @@ export class CoreBBCodeParser extends BBCodeParser { input = input.replace(findUrlRegex, (match, tag) => tag === undefined ? `[url]${match}[/url]` : match); return super.parseEverything(input); } -} \ No newline at end of file +} diff --git a/bbcode/interfaces.ts b/bbcode/interfaces.ts deleted file mode 100644 index b639dc1..0000000 --- a/bbcode/interfaces.ts +++ /dev/null @@ -1 +0,0 @@ -export const enum InlineDisplayMode {DISPLAY_ALL, DISPLAY_SFW, DISPLAY_NONE} \ No newline at end of file diff --git a/bbcode/parser.ts b/bbcode/parser.ts index 59aad3d..b0c6c60 100644 --- a/bbcode/parser.ts +++ b/bbcode/parser.ts @@ -189,9 +189,9 @@ export class BBCodeParser { if(parent !== undefined) parent.appendChild(document.createTextNode(input.substring(mark, selfAllowed ? tagStart : i + 1))); return i; - } else if(!selfAllowed) - return mark - 1; - else if(isAllowed(tagKey)) + } + if(!selfAllowed) return mark - 1; + if(isAllowed(tagKey)) this.warning(`Unexpected closing ${tagKey} tag. Needed ${self} tag instead.`); } else if(isAllowed(tagKey)) this.warning(`Found closing ${tagKey} tag that was never opened.`); } diff --git a/bbcode/standard.ts b/bbcode/standard.ts index faa6635..69cf91b 100644 --- a/bbcode/standard.ts +++ b/bbcode/standard.ts @@ -1,23 +1,15 @@ import Vue from 'vue'; import { BBCodeElement } from '../chat/bbcode'; -import {InlineImage} from '../interfaces'; -import { analyzeUrlTag, CoreBBCodeParser } from './core'; -import {InlineDisplayMode} from './interfaces'; +import {InlineDisplayMode, InlineImage} from '../interfaces'; +import * as Utils from '../site/utils'; +import {analyzeUrlTag, CoreBBCodeParser} from './core'; import {BBCodeCustomTag, BBCodeSimpleTag, BBCodeTextTag} from './parser'; import UrlTagView from './UrlTagView.vue'; -interface StandardParserSettings { - siteDomain: string - staticDomain: string - animatedIcons: boolean - inlineDisplayMode: InlineDisplayMode -} - const usernameRegex = /^[a-zA-Z0-9_\-\s]+$/; export class StandardBBCodeParser extends CoreBBCodeParser { - allowInlines = true; inlines: {[key: string]: InlineImage | undefined} | undefined; cleanup: Vue[] = []; @@ -29,12 +21,12 @@ export class StandardBBCodeParser extends CoreBBCodeParser { const el = this.createElement('img'); el.className = 'inline-image'; el.title = el.alt = inline.name; - el.src = `${this.settings.staticDomain}images/charinline/${p1}/${p2}/${inline.hash}.${inline.extension}`; + el.src = `${Utils.staticDomain}images/charinline/${p1}/${p2}/${inline.hash}.${inline.extension}`; outerEl.appendChild(el); return outerEl; } - constructor(public settings: StandardParserSettings) { + constructor() { super(); const hrTag = new BBCodeSimpleTag('hr', 'hr', [], []); hrTag.noClosingTag = true; @@ -114,7 +106,7 @@ export class StandardBBCodeParser extends CoreBBCodeParser { if(!usernameRegex.test(content)) return; const a = parser.createElement('a'); - a.href = `${this.settings.siteDomain}c/${content}`; + a.href = `${Utils.siteDomain}c/${content}`; a.target = '_blank'; a.className = 'character-link'; a.appendChild(document.createTextNode(content)); @@ -127,10 +119,10 @@ export class StandardBBCodeParser extends CoreBBCodeParser { if(!usernameRegex.test(content)) return; const a = parser.createElement('a'); - a.href = `${this.settings.siteDomain}c/${content}`; + a.href = `${Utils.siteDomain}c/${content}`; a.target = '_blank'; const img = parser.createElement('img'); - img.src = `${this.settings.staticDomain}images/avatar/${content.toLowerCase()}.png`; + img.src = `${Utils.staticDomain}images/avatar/${content.toLowerCase()}.png`; img.className = 'character-avatar icon'; img.title = img.alt = content; a.appendChild(img); @@ -144,10 +136,10 @@ export class StandardBBCodeParser extends CoreBBCodeParser { if(!usernameRegex.test(content)) return; let extension = '.gif'; - if(!this.settings.animatedIcons) + if(!Utils.settings.animateEicons) extension = '.png'; const img = parser.createElement('img'); - img.src = `${this.settings.staticDomain}images/eicon/${content.toLowerCase()}${extension}`; + img.src = `${Utils.staticDomain}images/eicon/${content.toLowerCase()}${extension}`; img.className = 'character-avatar icon'; img.title = img.alt = content; parent.appendChild(img); @@ -155,15 +147,11 @@ export class StandardBBCodeParser extends CoreBBCodeParser { })); this.addTag(new BBCodeTextTag('img', (p, parent, param, content) => { const parser = p; - if(!this.allowInlines) { - parser.warning('Inline images are not allowed here.'); - return undefined; - } if(typeof parser.inlines === 'undefined') { parser.warning('This page does not support inline images.'); return undefined; } - const displayMode = this.settings.inlineDisplayMode; + const displayMode = Utils.settings.inlineDisplayMode; if(!/^\d+$/.test(param)) { parser.warning('img tag parameters must be numbers.'); return undefined; @@ -228,9 +216,3 @@ export class StandardBBCodeParser extends CoreBBCodeParser { return elm; } } - -export let standardParser: StandardBBCodeParser; - -export function initParser(settings: StandardParserSettings): void { - standardParser = new StandardBBCodeParser(settings); -} \ No newline at end of file diff --git a/bbcode/view.ts b/bbcode/view.ts new file mode 100644 index 0000000..4f5bab4 --- /dev/null +++ b/bbcode/view.ts @@ -0,0 +1,26 @@ +import {CreateElement, FunctionalComponentOptions, RenderContext, VNode} from 'vue'; +import {DefaultProps, RecordPropsDefinition} from 'vue/types/options'; //tslint:disable-line:no-submodule-imports +import {BBCodeElement} from './core'; +import {BBCodeParser} from './parser'; + +export const BBCodeView = (parser: BBCodeParser): FunctionalComponentOptions> => ({ + functional: true, + render(createElement: CreateElement, context: RenderContext): VNode { + /*tslint:disable:no-unsafe-any*///because we're not actually supposed to do any of this + context.data.hook = { + insert(node: VNode): void { + node.elm!.appendChild(parser.parseEverything( + context.props.text !== undefined ? context.props.text : context.props.unsafeText)); + if(context.props.afterInsert !== undefined) context.props.afterInsert(node.elm); + }, + destroy(node: VNode): void { + const element = ((node.elm).firstChild); + if(element.cleanup !== undefined) element.cleanup(); + } + }; + const vnode = createElement('span', context.data); + vnode.key = context.props.text; + return vnode; + //tslint:enable + } +}); \ No newline at end of file diff --git a/chat/CharacterSearch.vue b/chat/CharacterSearch.vue index 5cbbca7..85d797c 100644 --- a/chat/CharacterSearch.vue +++ b/chat/CharacterSearch.vue @@ -38,10 +38,10 @@ \ No newline at end of file diff --git a/chat/ChatView.vue b/chat/ChatView.vue index ef041ab..e118962 100644 --- a/chat/ChatView.vue +++ b/chat/ChatView.vue @@ -33,7 +33,8 @@ @@ -55,7 +56,7 @@
+ @click.middle.prevent="conversation.close()"> {{conversation.name}} this.$refs['privateConversations'], { animation: 50, + fallbackTolerance: 5, onEnd: async(e) => { if(e.oldIndex === e.newIndex) return; return core.conversations.privateConversations[e.oldIndex!].sort(e.newIndex!); @@ -162,6 +164,7 @@ }); Sortable.create(this.$refs['channelConversations'], { animation: 50, + fallbackTolerance: 5, onEnd: async(e) => { if(e.oldIndex === e.newIndex) return; return core.conversations.channelConversations[e.oldIndex!].sort(e.newIndex!); @@ -268,13 +271,8 @@ overrideEl.id = 'overrideFontSize'; document.body.appendChild(overrideEl); const sheet = overrideEl.sheet; - const selectorList = ['#chatView', '.btn', '.form-control']; - for(const selector of selectorList) - sheet.insertRule(`${selector} { font-size: ${fontSize}px; }`, sheet.cssRules.length); - - const lineHeight = 1.428571429; - sheet.insertRule(`.form-control { line-height: ${lineHeight} }`, sheet.cssRules.length); - sheet.insertRule(`select.form-control { line-height: ${lineHeight} }`, sheet.cssRules.length); + sheet.insertRule(`#chatView, .btn, .form-control, .custom-select { font-size: ${fontSize}px; }`, sheet.cssRules.length); + sheet.insertRule(`.form-control, select.form-control { line-height: 1.428571429 }`, sheet.cssRules.length); } logOut(): void { @@ -441,6 +439,7 @@ #sidebar { .body a.btn { padding: 2px 0; + text-align: left; } @media (min-width: breakpoint-min(md)) { .sidebar { diff --git a/chat/ConversationView.vue b/chat/ConversationView.vue index fabc137..29eed3f 100644 --- a/chat/ConversationView.vue +++ b/chat/ConversationView.vue @@ -2,7 +2,7 @@
-
+
-
+
{{l('status.' + conversation.character.status)}}
@@ -158,10 +158,11 @@ import {Component, Hook, Prop, Watch} from '@f-list/vue-ts'; import Vue from 'vue'; import {EditorButton, EditorSelection} from '../bbcode/editor'; + import {BBCodeView} from '../bbcode/view'; import {isShowing as anyDialogsShown} from '../components/Modal.vue'; import {Keys} from '../keys'; import AdView from './AdView.vue'; - import {BBCodeView, Editor} from './bbcode'; + import {Editor} from './bbcode'; import CommandHelp from './CommandHelp.vue'; import { characterImage, getByteLength, getKey } from './common'; import ConversationSettings from './ConversationSettings.vue'; @@ -181,7 +182,7 @@ @Component({ components: { user: UserView, 'bbcode-editor': Editor, 'manage-channel': ManageChannel, settings: ConversationSettings, - logs: Logs, 'message-view': MessageView, bbcode: BBCodeView, 'command-help': CommandHelp, + logs: Logs, 'message-view': MessageView, bbcode: BBCodeView(core.bbCodeParser), 'command-help': CommandHelp, 'ad-view': AdView, 'channel-list': UserChannelList } }) @@ -326,8 +327,13 @@ if(this.messageView.scrollTop < 20) { if(!this.scrolledUp) { const firstMessage = this.messageView.firstElementChild; - if(this.conversation.loadMore() && firstMessage !== null) - this.$nextTick(() => setTimeout(() => this.messageView.scrollTop = (firstMessage).offsetTop, 0)); + if(this.conversation.loadMore() && firstMessage !== null) { + this.messageView.style.overflow = 'hidden'; + this.$nextTick(() => { + this.messageView.scrollTop = (firstMessage).offsetTop; + this.messageView.style.overflow = 'auto'; + }); + } } this.scrolledUp = true; } else this.scrolledUp = false; @@ -795,4 +801,4 @@ } } } - \ No newline at end of file + diff --git a/chat/Logs.vue b/chat/Logs.vue index 55a70cc..8b47028 100644 --- a/chat/Logs.vue +++ b/chat/Logs.vue @@ -85,7 +85,7 @@ components: {modal: Modal, 'message-view': MessageView, 'filterable-select': FilterableSelect} }) export default class Logs extends CustomDialog { - @Prop() + @Prop readonly conversation?: Conversation; conversations: LogInterface.Conversation[] = []; selectedConversation: LogInterface.Conversation | undefined; diff --git a/chat/ReportDialog.vue b/chat/ReportDialog.vue index adf3fd4..fdfe675 100644 --- a/chat/ReportDialog.vue +++ b/chat/ReportDialog.vue @@ -16,9 +16,10 @@ \ No newline at end of file diff --git a/components/FilterableSelect.vue b/components/FilterableSelect.vue index ad53985..d7239f4 100644 --- a/components/FilterableSelect.vue +++ b/components/FilterableSelect.vue @@ -1,11 +1,10 @@