From ebf7cb43c5e382d5caaa520c75c848cdbf3aa829 Mon Sep 17 00:00:00 2001 From: MayaWolf Date: Tue, 5 Dec 2017 02:47:27 +0100 Subject: [PATCH] 0.2.7 - Profile viewer and many many bug fixes --- bbcode/Editor.vue | 7 +- bbcode/core.ts | 13 +- bbcode/editor.ts | 2 +- bbcode/interfaces.ts | 1 + bbcode/parser.ts | 24 +- bbcode/standard.ts | 215 ++++++++++++++ chat/ChannelList.vue | 8 +- chat/CharacterSearch.vue | 3 +- chat/Chat.vue | 3 +- chat/ChatView.vue | 87 +++++- chat/CommandHelp.vue | 5 +- chat/ConversationView.vue | 21 +- chat/Logs.vue | 2 +- chat/SettingsView.vue | 11 +- chat/UserList.vue | 2 +- chat/UserMenu.vue | 22 +- chat/bbcode.ts | 4 +- chat/common.ts | 3 +- chat/conversations.ts | 56 ++-- chat/interfaces.ts | 5 +- chat/localize.ts | 14 +- chat/profile_api.ts | 172 +++++++++++ chat/user_view.ts | 4 +- chat/vue-raven.ts | 1 + components/FilterableSelect.vue | 2 +- components/Modal.vue | 2 +- components/character_link.vue | 35 +++ components/date_display.vue | 35 +++ components/form_errors.vue | 46 +++ cordova/Index.vue | 20 +- cordova/tsconfig.json | 2 - electron/Index.vue | 45 ++- electron/application.json | 2 +- electron/chat.ts | 3 + electron/filesystem.ts | 37 ++- electron/importer.ts | 4 +- electron/main.ts | 68 ++--- electron/menu.ts | 12 +- electron/package.json | 6 +- electron/tsconfig.json | 2 - electron/window_state.ts | 2 +- fchat/channels.ts | 32 +- fchat/characters.ts | 3 +- fchat/connection.ts | 19 +- fchat/interfaces.ts | 2 +- less/bbcode.less | 14 +- less/character_editor.less | 4 +- less/character_page.less | 255 +++++++++++----- less/chat.less | 8 + less/core.less | 18 +- less/flist_overrides.less | 6 + less/flist_variables.less | 30 +- less/themes/theme_base_chat.less | 1 + site/character_page/character_page.vue | 167 +++++++++++ site/character_page/contact_method.vue | 53 ++++ site/character_page/contact_utils.ts | 109 +++++++ site/character_page/context_menu.ts | 91 ++++++ site/character_page/copy_custom_dialog.vue | 100 +++++++ site/character_page/copy_custom_menu.vue | 47 +++ site/character_page/data_store.ts | 19 ++ site/character_page/delete_dialog.vue | 57 ++++ site/character_page/duplicate_dialog.vue | 108 +++++++ site/character_page/friend_dialog.vue | 189 ++++++++++++ site/character_page/friends.vue | 49 +++ site/character_page/groups.vue | 50 ++++ site/character_page/guestbook.vue | 143 +++++++++ site/character_page/guestbook_post.vue | 121 ++++++++ site/character_page/images.vue | 51 ++++ site/character_page/infotag.vue | 54 ++++ site/character_page/infotags.vue | 79 +++++ site/character_page/interfaces.ts | 327 +++++++++++++++++++++ site/character_page/kink.vue | 61 ++++ site/character_page/kinks.vue | 209 +++++++++++++ site/character_page/memo_dialog.vue | 71 +++++ site/character_page/report_dialog.vue | 141 +++++++++ site/character_page/sidebar.vue | 252 ++++++++++++++++ site/directives/vue-select.ts | 60 ++++ site/flash_display.ts | 93 ++++++ site/utils.ts | 111 +++++++ tslint.json | 15 +- tslint/noReturnAwaitRule.js | 36 --- 81 files changed, 3926 insertions(+), 337 deletions(-) create mode 100644 bbcode/interfaces.ts create mode 100644 bbcode/standard.ts create mode 100644 chat/profile_api.ts create mode 100644 components/character_link.vue create mode 100644 components/date_display.vue create mode 100644 components/form_errors.vue create mode 100644 site/character_page/character_page.vue create mode 100644 site/character_page/contact_method.vue create mode 100644 site/character_page/contact_utils.ts create mode 100644 site/character_page/context_menu.ts create mode 100644 site/character_page/copy_custom_dialog.vue create mode 100644 site/character_page/copy_custom_menu.vue create mode 100644 site/character_page/data_store.ts create mode 100644 site/character_page/delete_dialog.vue create mode 100644 site/character_page/duplicate_dialog.vue create mode 100644 site/character_page/friend_dialog.vue create mode 100644 site/character_page/friends.vue create mode 100644 site/character_page/groups.vue create mode 100644 site/character_page/guestbook.vue create mode 100644 site/character_page/guestbook_post.vue create mode 100644 site/character_page/images.vue create mode 100644 site/character_page/infotag.vue create mode 100644 site/character_page/infotags.vue create mode 100644 site/character_page/interfaces.ts create mode 100644 site/character_page/kink.vue create mode 100644 site/character_page/kinks.vue create mode 100644 site/character_page/memo_dialog.vue create mode 100644 site/character_page/report_dialog.vue create mode 100644 site/character_page/sidebar.vue create mode 100644 site/directives/vue-select.ts create mode 100644 site/flash_display.ts create mode 100644 site/utils.ts delete mode 100644 tslint/noReturnAwaitRule.js diff --git a/bbcode/Editor.vue b/bbcode/Editor.vue index 58c4301..31654af 100644 --- a/bbcode/Editor.vue +++ b/bbcode/Editor.vue @@ -34,6 +34,7 @@ import {getKey} from '../chat/common'; import {CoreBBCodeParser, urlRegex} from './core'; import {defaultButtons, EditorButton, EditorSelection} from './editor'; + import {BBCodeParser} from './parser'; @Component export default class Editor extends Vue { @@ -56,10 +57,14 @@ element: HTMLTextAreaElement; maxHeight: number; minHeight: number; - protected parser = new CoreBBCodeParser(); + protected parser: BBCodeParser; protected defaultButtons = defaultButtons; private isShiftPressed = false; + created(): void { + this.parser = new CoreBBCodeParser(); + } + mounted(): void { this.element = this.$refs['input']; const $element = $(this.element); diff --git a/bbcode/core.ts b/bbcode/core.ts index a65bc17..ba6d9dc 100644 --- a/bbcode/core.ts +++ b/bbcode/core.ts @@ -44,7 +44,7 @@ export class CoreBBCodeParser extends BBCodeParser { parent.appendChild(el); return el; }, (parser, element, _, param) => { - const content = element.innerText.trim(); + const content = element.textContent!.trim(); while(element.firstChild !== null) element.removeChild(element.firstChild); let url: string, display: string = content; @@ -54,23 +54,26 @@ export class CoreBBCodeParser extends BBCodeParser { } else if(content.length > 0) url = content; else { parser.warning('url tag contains no url.'); - element.innerText = ''; //Dafuq!? + element.textContent = ''; //Dafuq!? return; } // This fixes problems where content based urls are marked as invalid if they contain spaces. url = fixURL(url); if(!urlRegex.test(url)) { - element.innerText = `[BAD URL] ${url}`; + element.textContent = `[BAD URL] ${url}`; return; } + const fa = parser.createElement('i'); + fa.className = 'fa fa-link'; + element.appendChild(fa); const a = parser.createElement('a'); a.href = url; a.rel = 'nofollow noreferrer noopener'; a.target = '_blank'; - a.className = 'link-graphic'; + a.className = 'user-link'; a.title = url; - a.innerText = display; + a.textContent = display; element.appendChild(a); const span = document.createElement('span'); span.className = 'link-domain'; diff --git a/bbcode/editor.ts b/bbcode/editor.ts index e7536c7..e020a09 100644 --- a/bbcode/editor.ts +++ b/bbcode/editor.ts @@ -48,7 +48,7 @@ export let defaultButtons: ReadonlyArray = [ tag: 'color', startText: '[color=]', icon: 'fa-eyedropper', - key: 'q' + key: 'd' }, { title: 'Superscript (Ctrl+↑)\n\nLifts text above the text baseline. Makes text slightly smaller. Cannot be nested.', diff --git a/bbcode/interfaces.ts b/bbcode/interfaces.ts new file mode 100644 index 0000000..b639dc1 --- /dev/null +++ b/bbcode/interfaces.ts @@ -0,0 +1 @@ +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 3fef1d7..bcd1596 100644 --- a/bbcode/parser.ts +++ b/bbcode/parser.ts @@ -18,10 +18,10 @@ export abstract class BBCodeTag { } //tslint:disable-next-line:no-empty - afterClose(_: BBCodeParser, __: HTMLElement, ___: HTMLElement, ____?: string): void { + afterClose(_: BBCodeParser, __: HTMLElement, ___: HTMLElement | undefined, ____?: string): void { } - abstract createElement(parser: BBCodeParser, parent: HTMLElement, param: string): HTMLElement; + abstract createElement(parser: BBCodeParser, parent: HTMLElement, param: string): HTMLElement | undefined; } export class BBCodeSimpleTag extends BBCodeTag { @@ -33,8 +33,8 @@ export class BBCodeSimpleTag extends BBCodeTag { createElement(parser: BBCodeParser, parent: HTMLElement, param: string): HTMLElement { if(param.length > 0) parser.warning('Unexpected parameter'); - const el = parser.createElement(this.elementName); - if(this.classes !== undefined) + const el = parser.createElement(this.elementName); + if(this.classes !== undefined && this.classes.length > 0) el.className = this.classes.join(' '); parent.appendChild(el); /*tslint:disable-next-line:no-unsafe-any*/// false positive @@ -42,7 +42,7 @@ export class BBCodeSimpleTag extends BBCodeTag { } } -export type CustomElementCreator = (parser: BBCodeParser, parent: HTMLElement, param: string) => HTMLElement; +export type CustomElementCreator = (parser: BBCodeParser, parent: HTMLElement, param: string) => HTMLElement | undefined; export type CustomCloser = (parser: BBCodeParser, current: HTMLElement, parent: HTMLElement, param: string) => void; export class BBCodeCustomTag extends BBCodeTag { @@ -50,7 +50,7 @@ export class BBCodeCustomTag extends BBCodeTag { super(tag, tagList); } - createElement(parser: BBCodeParser, parent: HTMLElement, param: string): HTMLElement { + createElement(parser: BBCodeParser, parent: HTMLElement, param: string): HTMLElement | undefined { return this.customCreator(parser, parent, param); } @@ -63,7 +63,7 @@ export class BBCodeCustomTag extends BBCodeTag { enum BufferType { Raw, Tag } class ParserTag { - constructor(public tag: string, public param: string, public element: HTMLElement, public parent: HTMLElement, + constructor(public tag: string, public param: string, public element: HTMLElement, public parent: HTMLElement | undefined, public line: number, public column: number) { } @@ -155,8 +155,7 @@ export class BBCodeParser { let curType: BufferType = BufferType.Raw; // Root tag collects output. - const root = this.createElement('span'); - const rootTag = new ParserTag('', '', root, root, 1, 1); + const rootTag = new ParserTag('', '', this.createElement('span'), undefined, 1, 1); stack.push(rootTag); this._currentTag = rootTag; let paramStart = -1; @@ -207,13 +206,18 @@ export class BBCodeParser { if(!allowed) break; } + const tag = this._tags[tagKey]!; if(!allowed) { ignoreNextClosingTag(tagKey); quickReset(i); continue; } const parent = stackTop().element; - const el = this._tags[tagKey]!.createElement(this, parent, param); + const el: HTMLElement | undefined = tag.createElement(this, parent, param); + if(el === undefined) { + quickReset(i); + continue; + } if(!this._tags[tagKey]!.noClosingTag) stack.push(new ParserTag(tagKey, param, el, parent, this._line, this._column)); } else if(ignoreClosing[tagKey] > 0) { diff --git a/bbcode/standard.ts b/bbcode/standard.ts new file mode 100644 index 0000000..bb885ca --- /dev/null +++ b/bbcode/standard.ts @@ -0,0 +1,215 @@ +import * as $ from 'jquery'; +import {CoreBBCodeParser} from './core'; +import {InlineDisplayMode} from './interfaces'; +import {BBCodeCustomTag, BBCodeSimpleTag} from './parser'; + +interface InlineImage { + id: number + hash: string + extension: string + nsfw: boolean +} + +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; + + constructor(public settings: StandardParserSettings) { + super(); + const hrTag = new BBCodeSimpleTag('hr', 'hr', [], []); + hrTag.noClosingTag = true; + this.addTag('hr', hrTag); + this.addTag('quote', new BBCodeCustomTag('quote', (parser, parent, param) => { + if(param !== '') + parser.warning('Unexpected paramter on quote tag.'); + const element = parser.createElement('blockquote'); + const innerElement = parser.createElement('div'); + innerElement.className = 'quoteHeader'; + innerElement.appendChild(document.createTextNode('Quote:')); + element.appendChild(innerElement); + parent.appendChild(element); + return element; + })); + this.addTag('left', new BBCodeSimpleTag('left', 'span', ['leftText'])); + this.addTag('right', new BBCodeSimpleTag('right', 'span', ['rightText'])); + this.addTag('center', new BBCodeSimpleTag('center', 'span', ['centerText'])); + this.addTag('justify', new BBCodeSimpleTag('justify', 'span', ['justifyText'])); + this.addTag('big', new BBCodeSimpleTag('big', 'span', ['bigText'], ['url', 'i', 'u', 'b', 'color', 's'])); + this.addTag('small', new BBCodeSimpleTag('small', 'span', ['smallText'], ['url', 'i', 'u', 'b', 'color', 's'])); + this.addTag('indent', new BBCodeSimpleTag('indent', 'div', ['indentText'])); + this.addTag('heading', new BBCodeSimpleTag('heading', 'h2', [], ['url', 'i', 'u', 'b', 'color', 's'])); + this.addTag('collapse', new BBCodeCustomTag('collapse', (parser, parent, param) => { + if(param === '') { //tslint:disable-line:curly + parser.warning('title parameter is required.'); + // HACK: Compatability fix with old site. Titles are not trimmed on old site, so empty collapse titles need to be allowed. + //return null; + } + const outer = parser.createElement('div'); + outer.className = 'collapseHeader'; + const headerText = parser.createElement('div'); + headerText.className = 'collapseHeaderText'; + outer.appendChild(headerText); + const innerText = parser.createElement('span'); + innerText.appendChild(document.createTextNode(param)); + headerText.appendChild(innerText); + const body = parser.createElement('div'); + body.className = 'collapseBlock'; + outer.appendChild(body); + parent.appendChild(outer); + return body; + })); + this.addTag('user', new BBCodeCustomTag('user', (parser, parent, _) => { + const el = parser.createElement('span'); + parent.appendChild(el); + return el; + }, (parser, element, parent, param) => { + if(param !== '') + parser.warning('Unexpected parameter on user tag.'); + const content = element.innerText; + if(!usernameRegex.test(content)) + return; + const a = parser.createElement('a'); + a.href = `${this.settings.siteDomain}c/${content}`; + a.target = '_blank'; + a.className = 'character-link'; + a.appendChild(document.createTextNode(content)); + parent.replaceChild(a, element); + }, [])); + this.addTag('icon', new BBCodeCustomTag('icon', (parser, parent, _) => { + const el = parser.createElement('span'); + parent.appendChild(el); + return el; + }, (parser, element, parent, param) => { + if(param !== '') + parser.warning('Unexpected parameter on icon tag.'); + const content = element.innerText; + if(!usernameRegex.test(content)) + return; + const a = parser.createElement('a'); + a.href = `${this.settings.siteDomain}c/${content}`; + a.target = '_blank'; + const img = parser.createElement('img'); + img.src = `${this.settings.staticDomain}images/avatar/${content.toLowerCase()}.png`; + img.className = 'character-avatar icon'; + a.appendChild(img); + parent.replaceChild(a, element); + }, [])); + this.addTag('eicon', new BBCodeCustomTag('eicon', (parser, parent, _) => { + const el = parser.createElement('span'); + parent.appendChild(el); + return el; + }, (parser, element, parent, param) => { + if(param !== '') + parser.warning('Unexpected parameter on eicon tag.'); + const content = element.innerText; + + if(!usernameRegex.test(content)) + return; + let extension = '.gif'; + if(!this.settings.animatedIcons) + extension = '.png'; + const img = parser.createElement('img'); + img.src = `${this.settings.staticDomain}images/eicon/${content.toLowerCase()}${extension}`; + img.className = 'character-avatar icon'; + parent.replaceChild(img, element); + }, [])); + this.addTag('img', new BBCodeCustomTag('img', (p, parent, param) => { + 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; + } + let p1: string, p2: string, inline; + const displayMode = this.settings.inlineDisplayMode; + if(!/^\d+$/.test(param)) { + parser.warning('img tag parameters must be numbers.'); + return undefined; + } + if(typeof parser.inlines[param] !== 'object') { + parser.warning(`Could not find an inline image with id ${param} It will not be visible.`); + return undefined; + } + inline = parser.inlines[param]!; + p1 = inline.hash.substr(0, 2); + p2 = inline.hash.substr(2, 2); + + if(displayMode === InlineDisplayMode.DISPLAY_NONE || (displayMode === InlineDisplayMode.DISPLAY_SFW && inline.nsfw)) { + const el = parser.createElement('a'); + el.className = 'unloadedInline'; + el.href = '#'; + el.dataset.inlineId = param; + el.onclick = () => { + $('.unloadedInline').each((_, element) => { + const inlineId = $(element).data('inline-id'); + if(typeof parser.inlines![inlineId] !== 'object') + return; + const showInline = parser.inlines![inlineId]!; + const showP1 = showInline.hash.substr(0, 2); + const showP2 = showInline.hash.substr(2, 2); + //tslint:disable-next-line:max-line-length + $(element).replaceWith(`
`); + }); + return false; + }; + const prefix = inline.nsfw ? '[NSFW Inline] ' : '[Inline] '; + el.appendChild(document.createTextNode(prefix)); + parent.appendChild(el); + return el; + } else { + const outerEl = parser.createElement('div'); + const el = parser.createElement('img'); + el.className = 'imageBlock'; + el.src = `${this.settings.staticDomain}images/charinline/${p1}/${p2}/${inline.hash}.${inline.extension}`; + outerEl.appendChild(el); + parent.appendChild(outerEl); + return el; + } + }, (_, element, __, ___) => { + // Need to remove any appended contents, because this is a total hack job. + if(element.className !== 'imageBlock') + return; + while(element.firstChild !== null) + element.removeChild(element.firstChild); + }, [])); + } +} + +export function initCollapse(): void { + $('.collapseHeader[data-bound!=true]').each((_, element) => { + const $element = $(element); + const $body = $element.children('.collapseBlock'); + $element.children('.collapseHeaderText').on('click', () => { + if($element.hasClass('expandedHeader')) { + $body.css('max-height', '0'); + $element.removeClass('expandedHeader'); + } else { + $body.css('max-height', 'none'); + const height = $body.outerHeight(); + $body.css('max-height', '0'); + $element.addClass('expandedHeader'); + setTimeout(() => $body.css('max-height', height!), 1); + setTimeout(() => $body.css('max-height', 'none'), 250); + } + }); + }); + $('.collapseHeader').attr('data-bound', 'true'); +} + +export let standardParser: StandardBBCodeParser; + +export function initParser(settings: StandardParserSettings): void { + standardParser = new StandardBBCodeParser(settings); +} \ No newline at end of file diff --git a/chat/ChannelList.vue b/chat/ChannelList.vue index 4fcd6c8..40470c4 100644 --- a/chat/ChannelList.vue +++ b/chat/ChannelList.vue @@ -1,5 +1,5 @@