diff --git a/bbcode/core.ts b/bbcode/core.ts
index 766a7be..a0fd2b5 100644
--- a/bbcode/core.ts
+++ b/bbcode/core.ts
@@ -4,7 +4,7 @@ const urlFormat = '((?:https?|ftps?|irc)://[^\\s/$.?#"\']+\\.[^\\s"]+)';
export const findUrlRegex = new RegExp(`(\\[url[=\\]]\\s*)?${urlFormat}`, 'gi');
export const urlRegex = new RegExp(`^${urlFormat}$`);
-function domain(url: string): string | undefined {
+export function domain(url: string): string | undefined {
const pieces = urlRegex.exec(url);
if(pieces === null) return;
const match = pieces[1].match(/(?:(https?|ftps?|irc):)?\/\/(?:www.)?([^\/]+)/);
@@ -17,6 +17,39 @@ function fixURL(url: string): string {
return url.replace(/ /g, '%20');
}
+export function analyzeUrlTag(parser: BBCodeParser, param: string, content: string): {success: boolean, url?: string, domain?: string, textContent: string} {
+ let url: string | undefined, textContent: string = content;
+ let success = true;
+
+ if(param.length > 0) {
+ url = param.trim();
+ if(content.length === 0) textContent = param;
+ } else if(content.length > 0) url = content;
+ else {
+ parser.warning('url tag contains no url.');
+ textContent = '';
+ success = false;
+ }
+
+ if((success) && (url)) {
+ // This fixes problems where content based urls are marked as invalid if they contain spaces.
+ url = fixURL(url);
+
+ if (!urlRegex.test(url)) {
+ textContent = `[BAD URL] ${url}`;
+ success = false;
+ }
+ }
+
+ return {
+ success,
+ url,
+ textContent,
+ domain: url ? domain(url) : undefined
+ };
+}
+
+
export class CoreBBCodeParser extends BBCodeParser {
/*tslint:disable-next-line:typedef*///https://github.com/palantir/tslint/issues/711
constructor(public makeLinksClickable = true) {
@@ -40,41 +73,32 @@ export class CoreBBCodeParser extends BBCodeParser {
return el;
}));
this.addTag(new BBCodeTextTag('url', (parser, parent, param, content) => {
+ const tagData = analyzeUrlTag(parser, param, content);
const element = parser.createElement('span');
+
parent.appendChild(element);
- let url: string, display: string = content;
- if(param.length > 0) {
- url = param.trim();
- if(content.length === 0) display = param;
- } else if(content.length > 0) url = content;
- else {
- parser.warning('url tag contains no url.');
- element.textContent = '';
+ if (!tagData.success) {
+ element.textContent = tagData.textContent;
return;
}
- // This fixes problems where content based urls are marked as invalid if they contain spaces.
- url = fixURL(url);
- if(!urlRegex.test(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.href = tagData.url as string;
a.rel = 'nofollow noreferrer noopener';
a.target = '_blank';
a.className = 'user-link';
- a.title = url;
- a.textContent = display;
+ a.title = tagData.url as string;
+ a.textContent = tagData.textContent;
element.appendChild(a);
const span = document.createElement('span');
span.className = 'link-domain bbcode-pseudo';
- span.textContent = ` [${domain(url)}]`;
+ span.textContent = ` [${tagData.domain}]`;
element.appendChild(span);
+
return element;
}));
}
diff --git a/bbcode/parser.ts b/bbcode/parser.ts
index fe9a4ee..59aad3d 100644
--- a/bbcode/parser.ts
+++ b/bbcode/parser.ts
@@ -92,6 +92,10 @@ export class BBCodeParser {
this._tags[impl.tag] = impl;
}
+ getTag(tag: string): BBCodeTag | undefined {
+ return this._tags[tag];
+ }
+
removeTag(tag: string): void {
delete this._tags[tag];
}
diff --git a/chat/ConversationView.vue b/chat/ConversationView.vue
index 1f9172e..bc03297 100644
--- a/chat/ConversationView.vue
+++ b/chat/ConversationView.vue
@@ -136,6 +136,7 @@
+
@@ -150,6 +151,7 @@
import { characterImage, getByteLength, getKey } from "./common";
import ConversationSettings from './ConversationSettings.vue';
import core from './core';
+ import ImagePreview from './ImagePreview.vue';
import {Channel, channelModes, Character, Conversation, Settings} from './interfaces';
import l from './localize';
import Logs from './Logs.vue';
@@ -162,7 +164,8 @@
@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, 'command-help': CommandHelp,
+ 'image-preview': ImagePreview
}
})
export default class ConversationView extends Vue {
@@ -471,6 +474,7 @@
updateAutoPostingState();
}
+
hasSFC(message: Conversation.Message): message is Conversation.SFCMessage {
return (>message).sfc !== undefined;
}
diff --git a/chat/ImagePreview.vue b/chat/ImagePreview.vue
new file mode 100644
index 0000000..d65b187
--- /dev/null
+++ b/chat/ImagePreview.vue
@@ -0,0 +1,132 @@
+
+
+
+
+
+
+
+
diff --git a/chat/UrlTagView.vue b/chat/UrlTagView.vue
new file mode 100644
index 0000000..dd5a0e1
--- /dev/null
+++ b/chat/UrlTagView.vue
@@ -0,0 +1,58 @@
+
+
+
+ {{text}}
+ [{{domain}}]
+
+
+
+
diff --git a/chat/bbcode.ts b/chat/bbcode.ts
index a80665c..02d8050 100644
--- a/chat/bbcode.ts
+++ b/chat/bbcode.ts
@@ -1,5 +1,5 @@
import Vue, {Component, CreateElement, RenderContext, VNode} from 'vue';
-import {CoreBBCodeParser} from '../bbcode/core';
+import { CoreBBCodeParser, analyzeUrlTag } from '../bbcode/core';
//tslint:disable-next-line:match-default-export-name
import BaseEditor from '../bbcode/Editor.vue';
import {BBCodeTextTag} from '../bbcode/parser';
@@ -7,6 +7,7 @@ import ChannelView from './ChannelTagView.vue';
import {characterImage} from './common';
import core from './core';
import {Character} from './interfaces';
+import UrlView from './UrlTagView.vue';
import UserView from './user_view';
export const BBCodeView: Component = {
@@ -100,6 +101,27 @@ export default class BBCodeParser extends CoreBBCodeParser {
this.cleanup.push(view);
return root;
}));
+
+ this.addTag(new BBCodeTextTag(
+ 'url',
+ (parser, parent, _, content) => {
+ const tagData = analyzeUrlTag(parser, _, content);
+
+ const root = parser.createElement('span');
+ // const el = parser.createElement('span');
+ parent.appendChild(root);
+ // root.appendChild(el);
+
+ if (!tagData.success) {
+ root.textContent = tagData.textContent;
+ return;
+ }
+
+ const view = new UrlView({el: root, propsData: {url: tagData.url, text: tagData.textContent, domain: tagData.domain}});
+ this.cleanup.push(view);
+
+ return root;
+ }));
}
parseEverything(input: string): BBCodeElement {
diff --git a/chat/event-bus.ts b/chat/event-bus.ts
new file mode 100644
index 0000000..5b3e40d
--- /dev/null
+++ b/chat/event-bus.ts
@@ -0,0 +1,3 @@
+import Vue from 'vue';
+export const EventBus = new Vue();
+