mirror of
https://github.com/citizenfx/cfx-server-data.git
synced 2025-12-12 06:14:09 +01:00
chat: wip rework as chat2
This commit is contained in:
@@ -1,255 +0,0 @@
|
||||
window.APP = {
|
||||
template: '#app_template',
|
||||
name: 'app',
|
||||
data() {
|
||||
return {
|
||||
style: CONFIG.style,
|
||||
showInput: false,
|
||||
showWindow: false,
|
||||
shouldHide: true,
|
||||
backingSuggestions: [],
|
||||
removedSuggestions: [],
|
||||
templates: CONFIG.templates,
|
||||
message: '',
|
||||
messages: [],
|
||||
oldMessages: [],
|
||||
oldMessagesIndex: -1,
|
||||
tplBackups: [],
|
||||
msgTplBackups: []
|
||||
};
|
||||
},
|
||||
destroyed() {
|
||||
clearInterval(this.focusTimer);
|
||||
window.removeEventListener('message', this.listener);
|
||||
},
|
||||
mounted() {
|
||||
post('http://chat/loaded', JSON.stringify({}));
|
||||
this.listener = window.addEventListener('message', (event) => {
|
||||
const item = event.data || event.detail; //'detail' is for debuging via browsers
|
||||
if (this[item.type]) {
|
||||
this[item.type](item);
|
||||
}
|
||||
});
|
||||
},
|
||||
watch: {
|
||||
messages() {
|
||||
if (this.showWindowTimer) {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
}
|
||||
this.showWindow = true;
|
||||
this.resetShowWindowTimer();
|
||||
|
||||
const messagesObj = this.$refs.messages;
|
||||
this.$nextTick(() => {
|
||||
messagesObj.scrollTop = messagesObj.scrollHeight;
|
||||
});
|
||||
},
|
||||
},
|
||||
computed: {
|
||||
suggestions() {
|
||||
return this.backingSuggestions.filter((el) => this.removedSuggestions.indexOf(el.name) <= -1);
|
||||
},
|
||||
},
|
||||
methods: {
|
||||
ON_SCREEN_STATE_CHANGE({ shouldHide }) {
|
||||
this.shouldHide = shouldHide;
|
||||
},
|
||||
ON_OPEN() {
|
||||
this.showInput = true;
|
||||
this.showWindow = true;
|
||||
if (this.showWindowTimer) {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
}
|
||||
this.focusTimer = setInterval(() => {
|
||||
if (this.$refs.input) {
|
||||
this.$refs.input.focus();
|
||||
} else {
|
||||
clearInterval(this.focusTimer);
|
||||
}
|
||||
}, 100);
|
||||
},
|
||||
ON_MESSAGE({ message }) {
|
||||
this.messages.push(message);
|
||||
},
|
||||
ON_CLEAR() {
|
||||
this.messages = [];
|
||||
this.oldMessages = [];
|
||||
this.oldMessagesIndex = -1;
|
||||
},
|
||||
ON_SUGGESTION_ADD({ suggestion }) {
|
||||
const duplicateSuggestion = this.backingSuggestions.find(a => a.name == suggestion.name);
|
||||
if (duplicateSuggestion) {
|
||||
if(suggestion.help || suggestion.params) {
|
||||
duplicateSuggestion.help = suggestion.help || "";
|
||||
duplicateSuggestion.params = suggestion.params || [];
|
||||
}
|
||||
return;
|
||||
}
|
||||
if (!suggestion.params) {
|
||||
suggestion.params = []; //TODO Move somewhere else
|
||||
}
|
||||
this.backingSuggestions.push(suggestion);
|
||||
},
|
||||
ON_SUGGESTION_REMOVE({ name }) {
|
||||
if(this.removedSuggestions.indexOf(name) <= -1) {
|
||||
this.removedSuggestions.push(name);
|
||||
}
|
||||
},
|
||||
ON_TEMPLATE_ADD({ template }) {
|
||||
if (this.templates[template.id]) {
|
||||
this.warn(`Tried to add duplicate template '${template.id}'`)
|
||||
} else {
|
||||
this.templates[template.id] = template.html;
|
||||
}
|
||||
},
|
||||
ON_UPDATE_THEMES({ themes }) {
|
||||
this.removeThemes();
|
||||
|
||||
this.setThemes(themes);
|
||||
},
|
||||
removeThemes() {
|
||||
for (let i = 0; i < document.styleSheets.length; i++) {
|
||||
const styleSheet = document.styleSheets[i];
|
||||
const node = styleSheet.ownerNode;
|
||||
|
||||
if (node.getAttribute('data-theme')) {
|
||||
node.parentNode.removeChild(node);
|
||||
}
|
||||
}
|
||||
|
||||
this.tplBackups.reverse();
|
||||
|
||||
for (const [ elem, oldData ] of this.tplBackups) {
|
||||
elem.innerText = oldData;
|
||||
}
|
||||
|
||||
this.tplBackups = [];
|
||||
|
||||
this.msgTplBackups.reverse();
|
||||
|
||||
for (const [ id, oldData ] of this.msgTplBackups) {
|
||||
this.templates[id] = oldData;
|
||||
}
|
||||
|
||||
this.msgTplBackups = [];
|
||||
},
|
||||
setThemes(themes) {
|
||||
for (const [ id, data ] of Object.entries(themes)) {
|
||||
if (data.style) {
|
||||
const style = document.createElement('style');
|
||||
style.type = 'text/css';
|
||||
style.setAttribute('data-theme', id);
|
||||
style.appendChild(document.createTextNode(data.style));
|
||||
|
||||
document.head.appendChild(style);
|
||||
}
|
||||
|
||||
if (data.styleSheet) {
|
||||
const link = document.createElement('link');
|
||||
link.rel = 'stylesheet';
|
||||
link.type = 'text/css';
|
||||
link.href = data.baseUrl + data.styleSheet;
|
||||
link.setAttribute('data-theme', id);
|
||||
|
||||
document.head.appendChild(link);
|
||||
}
|
||||
|
||||
if (data.templates) {
|
||||
for (const [ tplId, tpl ] of Object.entries(data.templates)) {
|
||||
const elem = document.getElementById(tplId);
|
||||
|
||||
if (elem) {
|
||||
this.tplBackups.push([ elem, elem.innerText ]);
|
||||
elem.innerText = tpl;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.script) {
|
||||
const script = document.createElement('script');
|
||||
script.type = 'text/javascript';
|
||||
script.src = data.baseUrl + data.script;
|
||||
|
||||
document.head.appendChild(script);
|
||||
}
|
||||
|
||||
if (data.msgTemplates) {
|
||||
for (const [ tplId, tpl ] of Object.entries(data.msgTemplates)) {
|
||||
this.msgTplBackups.push([ tplId, this.templates[tplId] ]);
|
||||
this.templates[tplId] = tpl;
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
warn(msg) {
|
||||
this.messages.push({
|
||||
args: [msg],
|
||||
template: '^3<b>CHAT-WARN</b>: ^0{0}',
|
||||
});
|
||||
},
|
||||
clearShowWindowTimer() {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
},
|
||||
resetShowWindowTimer() {
|
||||
this.clearShowWindowTimer();
|
||||
this.showWindowTimer = setTimeout(() => {
|
||||
if (!this.showInput) {
|
||||
this.showWindow = false;
|
||||
}
|
||||
}, CONFIG.fadeTimeout);
|
||||
},
|
||||
keyUp() {
|
||||
this.resize();
|
||||
},
|
||||
keyDown(e) {
|
||||
if (e.which === 38 || e.which === 40) {
|
||||
e.preventDefault();
|
||||
this.moveOldMessageIndex(e.which === 38);
|
||||
} else if (e.which == 33) {
|
||||
var buf = document.getElementsByClassName('chat-messages')[0];
|
||||
buf.scrollTop = buf.scrollTop - 100;
|
||||
} else if (e.which == 34) {
|
||||
var buf = document.getElementsByClassName('chat-messages')[0];
|
||||
buf.scrollTop = buf.scrollTop + 100;
|
||||
}
|
||||
},
|
||||
moveOldMessageIndex(up) {
|
||||
if (up && this.oldMessages.length > this.oldMessagesIndex + 1) {
|
||||
this.oldMessagesIndex += 1;
|
||||
this.message = this.oldMessages[this.oldMessagesIndex];
|
||||
} else if (!up && this.oldMessagesIndex - 1 >= 0) {
|
||||
this.oldMessagesIndex -= 1;
|
||||
this.message = this.oldMessages[this.oldMessagesIndex];
|
||||
} else if (!up && this.oldMessagesIndex - 1 === -1) {
|
||||
this.oldMessagesIndex = -1;
|
||||
this.message = '';
|
||||
}
|
||||
},
|
||||
resize() {
|
||||
const input = this.$refs.input;
|
||||
input.style.height = '5px';
|
||||
input.style.height = `${input.scrollHeight + 2}px`;
|
||||
},
|
||||
send(e) {
|
||||
if(this.message !== '') {
|
||||
post('http://chat/chatResult', JSON.stringify({
|
||||
message: this.message,
|
||||
}));
|
||||
this.oldMessages.unshift(this.message);
|
||||
this.oldMessagesIndex = -1;
|
||||
this.hideInput();
|
||||
} else {
|
||||
this.hideInput(true);
|
||||
}
|
||||
},
|
||||
hideInput(canceled = false) {
|
||||
if (canceled) {
|
||||
post('http://chat/chatResult', JSON.stringify({ canceled }));
|
||||
}
|
||||
this.message = '';
|
||||
this.showInput = false;
|
||||
clearInterval(this.focusTimer);
|
||||
this.resetShowWindowTimer();
|
||||
},
|
||||
},
|
||||
};
|
||||
434
resources/[gameplay]/chat/html/App.ts
Normal file
434
resources/[gameplay]/chat/html/App.ts
Normal file
@@ -0,0 +1,434 @@
|
||||
import { post } from './utils';
|
||||
import CONFIG from './config';
|
||||
import Vue from 'vue';
|
||||
|
||||
import Suggestions from './Suggestions.vue';
|
||||
import MessageV from './Message.vue';
|
||||
import { Suggestion } from './Suggestions';
|
||||
|
||||
export interface Message {
|
||||
args: string[];
|
||||
template: string;
|
||||
params?: { [key: string]: string };
|
||||
multiline?: boolean;
|
||||
color?: [ number, number, number ];
|
||||
templateId?: number;
|
||||
|
||||
id?: string;
|
||||
}
|
||||
|
||||
export interface ThemeData {
|
||||
style: string;
|
||||
styleSheet: string;
|
||||
baseUrl: string;
|
||||
script: string;
|
||||
templates: { [id: string]: string }; // not supported rn
|
||||
msgTemplates: { [id: string]: string };
|
||||
|
||||
}
|
||||
|
||||
export interface Mode {
|
||||
name: string;
|
||||
displayName: string;
|
||||
color: string;
|
||||
}
|
||||
|
||||
enum ChatHideStates {
|
||||
ShowWhenActive = 0,
|
||||
AlwaysShow = 1,
|
||||
AlwaysHide = 2,
|
||||
}
|
||||
|
||||
const defaultMode: Mode = {
|
||||
name: 'all',
|
||||
displayName: 'All',
|
||||
color: '#fff'
|
||||
};
|
||||
|
||||
export default Vue.extend({
|
||||
template: "#app_template",
|
||||
name: "app",
|
||||
components: {
|
||||
Suggestions,
|
||||
MessageV
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
style: CONFIG.style,
|
||||
showInput: false,
|
||||
showWindow: false,
|
||||
showHideState: false,
|
||||
hideState: ChatHideStates.ShowWhenActive,
|
||||
backingSuggestions: [] as Suggestion[],
|
||||
removedSuggestions: [] as string[],
|
||||
templates: { ...CONFIG.templates } as { [ key: string ]: string },
|
||||
message: "",
|
||||
messages: [] as Message[],
|
||||
oldMessages: [] as string[],
|
||||
oldMessagesIndex: -1,
|
||||
tplBackups: [] as unknown as [ HTMLElement, string ][],
|
||||
msgTplBackups: [] as unknown as [ string, string ][],
|
||||
focusTimer: 0,
|
||||
showWindowTimer: 0,
|
||||
showHideStateTimer: 0,
|
||||
listener: (event: MessageEvent) => {},
|
||||
modes: [defaultMode] as Mode[],
|
||||
modeIdx: 0,
|
||||
};
|
||||
},
|
||||
destroyed() {
|
||||
clearInterval(this.focusTimer);
|
||||
window.removeEventListener("message", this.listener);
|
||||
},
|
||||
mounted() {
|
||||
post("http://chat/loaded", JSON.stringify({}));
|
||||
|
||||
this.listener = (event: MessageEvent) => {
|
||||
const item: any = event.data || (<any>event).detail; //'detail' is for debugging via browsers
|
||||
|
||||
if (!item || !item.type) {
|
||||
return;
|
||||
}
|
||||
|
||||
const typeRef = item.type as
|
||||
'ON_OPEN' | 'ON_SCREEN_STATE_CHANGE' | 'ON_MESSAGE' | 'ON_CLEAR' | 'ON_SUGGESTION_ADD' |
|
||||
'ON_SUGGESTION_REMOVE' | 'ON_TEMPLATE_ADD' | 'ON_UPDATE_THEMES' | 'ON_MODE_ADD' | 'ON_MODE_REMOVE';
|
||||
|
||||
if (this[typeRef]) {
|
||||
this[typeRef](item);
|
||||
}
|
||||
};
|
||||
|
||||
window.addEventListener("message", this.listener);
|
||||
},
|
||||
watch: {
|
||||
messages() {
|
||||
if (this.hideState !== ChatHideStates.AlwaysHide) {
|
||||
if (this.showWindowTimer) {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
}
|
||||
this.showWindow = true;
|
||||
this.resetShowWindowTimer();
|
||||
}
|
||||
|
||||
const messagesObj = this.$refs.messages as HTMLDivElement;
|
||||
this.$nextTick(() => {
|
||||
messagesObj.scrollTop = messagesObj.scrollHeight;
|
||||
});
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
suggestions(): Suggestion[] {
|
||||
return this.backingSuggestions.filter(
|
||||
el => this.removedSuggestions.indexOf(el.name) <= -1
|
||||
);
|
||||
},
|
||||
|
||||
hideAnimated(): boolean {
|
||||
return this.hideState !== ChatHideStates.AlwaysHide;
|
||||
},
|
||||
|
||||
modeIdxGet(): number {
|
||||
return (this.modeIdx >= this.modes.length) ? (this.modes.length - 1) : this.modeIdx;
|
||||
},
|
||||
|
||||
modePrefix(): string {
|
||||
if (this.modes.length === 1) {
|
||||
return `➤`;
|
||||
}
|
||||
|
||||
return this.modes[this.modeIdxGet].displayName;
|
||||
},
|
||||
|
||||
modeColor(): string {
|
||||
return this.modes[this.modeIdxGet].color;
|
||||
},
|
||||
|
||||
hideStateString(): string {
|
||||
// TODO: localization
|
||||
switch (this.hideState) {
|
||||
case ChatHideStates.AlwaysShow:
|
||||
return 'Visible';
|
||||
case ChatHideStates.AlwaysHide:
|
||||
return 'Hidden';
|
||||
case ChatHideStates.ShowWhenActive:
|
||||
return 'When active';
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
ON_SCREEN_STATE_CHANGE({ hideState, fromUserInteraction }: { hideState: ChatHideStates, fromUserInteraction: boolean }) {
|
||||
this.hideState = hideState;
|
||||
|
||||
if (this.hideState === ChatHideStates.AlwaysHide) {
|
||||
if (!this.showInput) {
|
||||
this.showWindow = false;
|
||||
}
|
||||
} else if (this.hideState === ChatHideStates.AlwaysShow) {
|
||||
this.showWindow = true;
|
||||
if (this.showWindowTimer) {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
}
|
||||
} else {
|
||||
this.resetShowWindowTimer();
|
||||
}
|
||||
|
||||
if (fromUserInteraction) {
|
||||
this.showHideState = true;
|
||||
|
||||
if (this.showHideStateTimer) {
|
||||
clearTimeout(this.showHideStateTimer);
|
||||
}
|
||||
|
||||
this.showHideStateTimer = window.setTimeout(() => {
|
||||
this.showHideState = false;
|
||||
}, 1500);
|
||||
}
|
||||
},
|
||||
ON_OPEN() {
|
||||
this.showInput = true;
|
||||
this.showWindow = true;
|
||||
if (this.showWindowTimer) {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
}
|
||||
this.focusTimer = window.setInterval(() => {
|
||||
if (this.$refs.input) {
|
||||
(this.$refs.input as HTMLInputElement).focus();
|
||||
} else {
|
||||
clearInterval(this.focusTimer);
|
||||
}
|
||||
}, 100);
|
||||
},
|
||||
ON_MESSAGE({ message }: { message: Message }) {
|
||||
message.id = `${new Date().getTime()}${Math.random()}`;
|
||||
this.messages.push(message);
|
||||
},
|
||||
ON_CLEAR() {
|
||||
this.messages = [];
|
||||
this.oldMessages = [];
|
||||
this.oldMessagesIndex = -1;
|
||||
},
|
||||
ON_SUGGESTION_ADD({ suggestion }: { suggestion: Suggestion }) {
|
||||
const duplicateSuggestion = this.backingSuggestions.find(
|
||||
a => a.name == suggestion.name
|
||||
);
|
||||
if (duplicateSuggestion) {
|
||||
if (suggestion.help || suggestion.params) {
|
||||
duplicateSuggestion.help = suggestion.help || "";
|
||||
duplicateSuggestion.params = suggestion.params || [];
|
||||
}
|
||||
return;
|
||||
}
|
||||
if (!suggestion.params) {
|
||||
suggestion.params = []; //TODO Move somewhere else
|
||||
}
|
||||
this.backingSuggestions.push(suggestion);
|
||||
},
|
||||
ON_SUGGESTION_REMOVE({ name }: { name: string }) {
|
||||
if (this.removedSuggestions.indexOf(name) <= -1) {
|
||||
this.removedSuggestions.push(name);
|
||||
}
|
||||
},
|
||||
ON_MODE_ADD({ mode }: { mode: Mode }) {
|
||||
this.modes = [
|
||||
...this.modes.filter(a => a.name !== mode.name),
|
||||
mode
|
||||
];
|
||||
},
|
||||
ON_MODE_REMOVE({ name }: { name: string }) {
|
||||
this.modes = this.modes.filter(a => a.name !== name);
|
||||
|
||||
if (this.modes.length === 0) {
|
||||
this.modes = [defaultMode];
|
||||
}
|
||||
},
|
||||
ON_TEMPLATE_ADD({ template }: { template: { id: string, html: string }}) {
|
||||
if (this.templates[template.id]) {
|
||||
this.warn(`Tried to add duplicate template '${template.id}'`);
|
||||
} else {
|
||||
this.templates[template.id] = template.html;
|
||||
}
|
||||
},
|
||||
ON_UPDATE_THEMES({ themes }: { themes: { [key: string]: ThemeData } }) {
|
||||
this.removeThemes();
|
||||
|
||||
this.setThemes(themes);
|
||||
},
|
||||
removeThemes() {
|
||||
for (let i = 0; i < document.styleSheets.length; i++) {
|
||||
const styleSheet = document.styleSheets[i];
|
||||
const node = styleSheet.ownerNode as Element;
|
||||
|
||||
if (node.getAttribute("data-theme")) {
|
||||
node.parentNode?.removeChild(node);
|
||||
}
|
||||
}
|
||||
|
||||
this.tplBackups.reverse();
|
||||
|
||||
for (const [elem, oldData] of this.tplBackups) {
|
||||
elem.innerText = oldData;
|
||||
}
|
||||
|
||||
this.tplBackups = [];
|
||||
|
||||
this.msgTplBackups.reverse();
|
||||
|
||||
for (const [id, oldData] of this.msgTplBackups) {
|
||||
this.templates[id] = oldData;
|
||||
}
|
||||
|
||||
this.msgTplBackups = [];
|
||||
},
|
||||
setThemes(themes: { [key: string]: ThemeData }) {
|
||||
for (const [id, data] of Object.entries(themes)) {
|
||||
if (data.style) {
|
||||
const style = document.createElement("style");
|
||||
style.type = "text/css";
|
||||
style.setAttribute("data-theme", id);
|
||||
style.appendChild(document.createTextNode(data.style));
|
||||
|
||||
document.head.appendChild(style);
|
||||
}
|
||||
|
||||
if (data.styleSheet) {
|
||||
const link = document.createElement("link");
|
||||
link.rel = "stylesheet";
|
||||
link.type = "text/css";
|
||||
link.href = data.baseUrl + data.styleSheet;
|
||||
link.setAttribute("data-theme", id);
|
||||
|
||||
document.head.appendChild(link);
|
||||
}
|
||||
|
||||
if (data.templates) {
|
||||
for (const [tplId, tpl] of Object.entries(data.templates)) {
|
||||
const elem = document.getElementById(tplId);
|
||||
|
||||
if (elem) {
|
||||
this.tplBackups.push([elem, elem.innerText]);
|
||||
elem.innerText = tpl;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (data.script) {
|
||||
const script = document.createElement("script");
|
||||
script.type = "text/javascript";
|
||||
script.src = data.baseUrl + data.script;
|
||||
|
||||
document.head.appendChild(script);
|
||||
}
|
||||
|
||||
if (data.msgTemplates) {
|
||||
for (const [tplId, tpl] of Object.entries(data.msgTemplates)) {
|
||||
this.msgTplBackups.push([tplId, this.templates[tplId]]);
|
||||
this.templates[tplId] = tpl;
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
warn(msg: string) {
|
||||
this.messages.push({
|
||||
args: [msg],
|
||||
template: "^3<b>CHAT-WARN</b>: ^0{0}"
|
||||
});
|
||||
},
|
||||
clearShowWindowTimer() {
|
||||
clearTimeout(this.showWindowTimer);
|
||||
},
|
||||
resetShowWindowTimer() {
|
||||
this.clearShowWindowTimer();
|
||||
this.showWindowTimer = window.setTimeout(() => {
|
||||
if (this.hideState !== ChatHideStates.AlwaysShow && !this.showInput) {
|
||||
this.showWindow = false;
|
||||
}
|
||||
}, CONFIG.fadeTimeout);
|
||||
},
|
||||
keyUp() {
|
||||
this.resize();
|
||||
},
|
||||
keyDown(e: KeyboardEvent) {
|
||||
if (e.which === 38 || e.which === 40) {
|
||||
e.preventDefault();
|
||||
this.moveOldMessageIndex(e.which === 38);
|
||||
} else if (e.which == 33) {
|
||||
var buf = document.getElementsByClassName("chat-messages")[0];
|
||||
buf.scrollTop = buf.scrollTop - 100;
|
||||
} else if (e.which == 34) {
|
||||
var buf = document.getElementsByClassName("chat-messages")[0];
|
||||
buf.scrollTop = buf.scrollTop + 100;
|
||||
} else if (e.which === 9) { // tab
|
||||
if (e.shiftKey || e.altKey) {
|
||||
--this.modeIdx;
|
||||
|
||||
if (this.modeIdx < 0) {
|
||||
this.modeIdx = this.modes.length - 1;
|
||||
}
|
||||
} else {
|
||||
this.modeIdx = (this.modeIdx + 1) % this.modes.length;
|
||||
}
|
||||
}
|
||||
|
||||
this.resize();
|
||||
},
|
||||
moveOldMessageIndex(up: boolean) {
|
||||
if (up && this.oldMessages.length > this.oldMessagesIndex + 1) {
|
||||
this.oldMessagesIndex += 1;
|
||||
this.message = this.oldMessages[this.oldMessagesIndex];
|
||||
} else if (!up && this.oldMessagesIndex - 1 >= 0) {
|
||||
this.oldMessagesIndex -= 1;
|
||||
this.message = this.oldMessages[this.oldMessagesIndex];
|
||||
} else if (!up && this.oldMessagesIndex - 1 === -1) {
|
||||
this.oldMessagesIndex = -1;
|
||||
this.message = "";
|
||||
}
|
||||
},
|
||||
resize() {
|
||||
const input = this.$refs.input as HTMLInputElement;
|
||||
|
||||
// scrollHeight includes padding, but content-box excludes padding
|
||||
// remove padding before setting height on the element
|
||||
const style = getComputedStyle(input);
|
||||
const paddingRemove = parseFloat(style.paddingBottom) + parseFloat(style.paddingTop);
|
||||
|
||||
input.style.height = "5px";
|
||||
input.style.height = `${input.scrollHeight - paddingRemove}px`;
|
||||
},
|
||||
send() {
|
||||
if (this.message !== "") {
|
||||
post(
|
||||
"http://chat/chatResult",
|
||||
JSON.stringify({
|
||||
message: this.message,
|
||||
mode: this.modes[this.modeIdxGet].name
|
||||
})
|
||||
);
|
||||
this.oldMessages.unshift(this.message);
|
||||
this.oldMessagesIndex = -1;
|
||||
this.hideInput();
|
||||
} else {
|
||||
this.hideInput(true);
|
||||
}
|
||||
},
|
||||
hideInput(canceled = false) {
|
||||
setTimeout(() => {
|
||||
const input = this.$refs.input as HTMLInputElement;
|
||||
delete input.style.height;
|
||||
}, 50);
|
||||
|
||||
if (canceled) {
|
||||
post("http://chat/chatResult", JSON.stringify({ canceled }));
|
||||
}
|
||||
this.message = "";
|
||||
this.showInput = false;
|
||||
clearInterval(this.focusTimer);
|
||||
|
||||
if (this.hideState !== ChatHideStates.AlwaysHide) {
|
||||
this.resetShowWindowTimer();
|
||||
} else {
|
||||
this.showWindow = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
44
resources/[gameplay]/chat/html/App.vue
Normal file
44
resources/[gameplay]/chat/html/App.vue
Normal file
@@ -0,0 +1,44 @@
|
||||
<template>
|
||||
<div id="app">
|
||||
<div class="chat-window" :style="this.style" :class="{
|
||||
'animated': !showWindow && hideAnimated,
|
||||
'hidden': !showWindow
|
||||
}">
|
||||
<div class="chat-messages" ref="messages">
|
||||
<message v-for="msg in messages"
|
||||
:templates="templates"
|
||||
:multiline="msg.multiline"
|
||||
:args="msg.args"
|
||||
:params="msg.params"
|
||||
:color="msg.color"
|
||||
:template="msg.template"
|
||||
:template-id="msg.templateId"
|
||||
:key="msg.id">
|
||||
</message>
|
||||
</div>
|
||||
</div>
|
||||
<div class="chat-input">
|
||||
<div v-show="showInput" class="input">
|
||||
<span class="prefix" :class="{ any: modes.length > 1 }" :style="{ color: modeColor }">{{modePrefix}}</span>
|
||||
<textarea v-model="message"
|
||||
ref="input"
|
||||
type="text"
|
||||
autofocus
|
||||
spellcheck="false"
|
||||
rows="1"
|
||||
@keyup.esc="hideInput"
|
||||
@keyup="keyUp"
|
||||
@keydown="keyDown"
|
||||
@keypress.enter.prevent="send">
|
||||
</textarea>
|
||||
</div>
|
||||
<suggestions :message="message" :suggestions="suggestions">
|
||||
</suggestions>
|
||||
<div class="chat-hide-state" v-show="showHideState">
|
||||
Chat: {{hideStateString}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script lang="ts" src="./App.ts"></script>
|
||||
@@ -1,42 +1,48 @@
|
||||
Vue.component('message', {
|
||||
template: '#message_template',
|
||||
import CONFIG from './config';
|
||||
import Vue, { PropType } from 'vue';
|
||||
|
||||
export default Vue.component('message', {
|
||||
data() {
|
||||
return {};
|
||||
},
|
||||
computed: {
|
||||
textEscaped() {
|
||||
textEscaped(): string {
|
||||
let s = this.template ? this.template : this.templates[this.templateId];
|
||||
|
||||
if (this.template) {
|
||||
//We disable templateId since we are using a direct raw template
|
||||
this.templateId = -1;
|
||||
}
|
||||
|
||||
//This hack is required to preserve backwards compatability
|
||||
if (this.templateId == CONFIG.defaultTemplateId
|
||||
if (!this.template && this.templateId == CONFIG.defaultTemplateId
|
||||
&& this.args.length == 1) {
|
||||
s = this.templates[CONFIG.defaultAltTemplateId] //Swap out default template :/
|
||||
}
|
||||
|
||||
s = s.replace(`@default`, this.templates[this.templateId]);
|
||||
|
||||
s = s.replace(/{(\d+)}/g, (match, number) => {
|
||||
const argEscaped = this.args[number] != undefined ? this.escape(this.args[number]) : match
|
||||
const argEscaped = this.args[number] != undefined ? this.escape(this.args[number]) : match;
|
||||
if (number == 0 && this.color) {
|
||||
//color is deprecated, use templates or ^1 etc.
|
||||
return this.colorizeOld(argEscaped);
|
||||
}
|
||||
return argEscaped;
|
||||
});
|
||||
|
||||
// format variant args
|
||||
s = s.replace(/\{\{([a-zA-Z0-9_\-]+?)\}\}/g, (match, id) => {
|
||||
const argEscaped = this.params[id] != undefined ? this.escape(this.params[id]) : match;
|
||||
return argEscaped;
|
||||
});
|
||||
|
||||
return this.colorize(s);
|
||||
},
|
||||
},
|
||||
methods: {
|
||||
colorizeOld(str) {
|
||||
colorizeOld(str: string): string {
|
||||
return `<span style="color: rgb(${this.color[0]}, ${this.color[1]}, ${this.color[2]})">${str}</span>`
|
||||
},
|
||||
colorize(str) {
|
||||
let s = "<span>" + (str.replace(/\^([0-9])/g, (str, color) => `</span><span class="color-${color}">`)) + "</span>";
|
||||
colorize(str: string): string {
|
||||
let s = "<span>" + colorTrans(str) + "</span>";
|
||||
|
||||
const styleDict = {
|
||||
const styleDict: {[ key: string ]: string} = {
|
||||
'*': 'font-weight: bold;',
|
||||
'_': 'text-decoration: underline;',
|
||||
'~': 'text-decoration: line-through;',
|
||||
@@ -49,8 +55,15 @@ Vue.component('message', {
|
||||
s = s.replace(styleRegex, (str, style, inner) => `<em style="${styleDict[style]}">${inner}</em>`)
|
||||
}
|
||||
return s.replace(/<span[^>]*><\/span[^>]*>/g, '');
|
||||
|
||||
function colorTrans(str: string) {
|
||||
return str
|
||||
.replace(/\^([0-9])/g, (str, color) => `</span><span class="color-${color}">`)
|
||||
.replace(/\^#([0-9A-F]{3,6})/gi, (str, color) => `</span><span class="color" style="color: #${color}">`)
|
||||
.replace(/~([a-z])~/g, (str, color) => `</span><span class="gameColor-${color}">`);
|
||||
}
|
||||
},
|
||||
escape(unsafe) {
|
||||
escape(unsafe: string): string {
|
||||
return String(unsafe)
|
||||
.replace(/&/g, '&')
|
||||
.replace(/</g, '<')
|
||||
@@ -61,10 +74,13 @@ Vue.component('message', {
|
||||
},
|
||||
props: {
|
||||
templates: {
|
||||
type: Object,
|
||||
type: Object as PropType<{ [key: string]: string }>,
|
||||
},
|
||||
args: {
|
||||
type: Array,
|
||||
type: Array as PropType<string[]>,
|
||||
},
|
||||
params: {
|
||||
type: Object as PropType<{ [ key: string]: string }>,
|
||||
},
|
||||
template: {
|
||||
type: String,
|
||||
@@ -79,8 +95,8 @@ Vue.component('message', {
|
||||
default: false,
|
||||
},
|
||||
color: { //deprecated
|
||||
type: Array,
|
||||
default: false,
|
||||
type: Array as PropType<number[]>,
|
||||
default: null,
|
||||
},
|
||||
},
|
||||
});
|
||||
7
resources/[gameplay]/chat/html/Message.vue
Normal file
7
resources/[gameplay]/chat/html/Message.vue
Normal file
@@ -0,0 +1,7 @@
|
||||
<template>
|
||||
<div class="msg" :class="{ multiline }">
|
||||
<span v-html="textEscaped"></span>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script lang="ts" src="./Message.ts"></script>
|
||||
@@ -1,11 +1,29 @@
|
||||
Vue.component('suggestions', {
|
||||
template: '#suggestions_template',
|
||||
props: ['message', 'suggestions'],
|
||||
import CONFIG from './config';
|
||||
import Vue, { PropType } from 'vue';
|
||||
|
||||
export interface Suggestion {
|
||||
name: string;
|
||||
help: string;
|
||||
params: string[];
|
||||
|
||||
disabled: boolean;
|
||||
}
|
||||
|
||||
export default Vue.component('suggestions', {
|
||||
props: {
|
||||
message: {
|
||||
type: String
|
||||
},
|
||||
|
||||
suggestions: {
|
||||
type: Array as PropType<Suggestion[]>
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {};
|
||||
},
|
||||
computed: {
|
||||
currentSuggestions() {
|
||||
currentSuggestions(): Suggestion[] {
|
||||
if (this.message === '') {
|
||||
return [];
|
||||
}
|
||||
@@ -34,6 +52,7 @@ Vue.component('suggestions', {
|
||||
const regex = new RegExp(`${s.name} (?:\\w+ ){${index}}(?:${wType}*)$`, 'g');
|
||||
|
||||
// eslint-disable-next-line no-param-reassign
|
||||
// @ts-ignore
|
||||
p.disabled = this.message.match(regex) == null;
|
||||
});
|
||||
});
|
||||
29
resources/[gameplay]/chat/html/Suggestions.vue
Normal file
29
resources/[gameplay]/chat/html/Suggestions.vue
Normal file
@@ -0,0 +1,29 @@
|
||||
<template>
|
||||
<div class="suggestions-wrap" v-show="currentSuggestions.length > 0">
|
||||
<ul class="suggestions">
|
||||
<li class="suggestion" v-for="s in currentSuggestions" :key="s.name">
|
||||
<p>
|
||||
<span :class="{ 'disabled': s.disabled }">
|
||||
{{s.name}}
|
||||
</span>
|
||||
<span class="param"
|
||||
v-for="p in s.params"
|
||||
:class="{ 'disabled': p.disabled }"
|
||||
:key="p.name">
|
||||
[{{p.name}}]
|
||||
</span>
|
||||
</p>
|
||||
<small class="help">
|
||||
<template v-if="!s.disabled">
|
||||
{{s.help}}
|
||||
</template>
|
||||
<template v-for="p in s.params" v-if="!p.disabled">
|
||||
{{p.help}}
|
||||
</template>
|
||||
</small>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script lang="ts" src="./Suggestions.ts"></script>
|
||||
@@ -1,6 +1,4 @@
|
||||
// DO NOT EDIT THIS FILE
|
||||
// Copy it to `config.js` and edit it
|
||||
window.CONFIG = {
|
||||
export default {
|
||||
defaultTemplateId: 'default', //This is the default template for 2 args1
|
||||
defaultAltTemplateId: 'defaultAlt', //This one for 1 arg
|
||||
templates: { //You can add static templates here
|
||||
@@ -13,7 +11,7 @@ window.CONFIG = {
|
||||
suggestionLimit: 5,
|
||||
style: {
|
||||
background: 'rgba(52, 73, 94, 0.7)',
|
||||
width: '38%',
|
||||
width: '38vw',
|
||||
height: '22%',
|
||||
}
|
||||
};
|
||||
@@ -8,6 +8,14 @@
|
||||
.color-8{color: #cc0000;}
|
||||
.color-9{color: #cc0068;}
|
||||
|
||||
.gameColor-w{color: #ffffff;}
|
||||
.gameColor-r{color: #ff4444;}
|
||||
.gameColor-g{color: #99cc00;}
|
||||
.gameColor-y{color: #ffbb33;}
|
||||
.gameColor-b{color: #33b5e5;}
|
||||
|
||||
/* todo: more game colors */
|
||||
|
||||
* {
|
||||
font-family: 'Lato', sans-serif;
|
||||
margin: 0;
|
||||
@@ -63,26 +71,47 @@ em {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
.chat-input > div.input {
|
||||
position: relative;
|
||||
display: flex;
|
||||
align-items: stretch;
|
||||
width: 100%;
|
||||
background-color: rgba(44, 62, 80, 1.0);
|
||||
}
|
||||
|
||||
.chat-hide-state {
|
||||
text-transform: uppercase;
|
||||
margin-left: 0.05vw;
|
||||
font-size: 1.65vh;
|
||||
}
|
||||
|
||||
.prefix {
|
||||
font-size: 1.8vh;
|
||||
position: absolute;
|
||||
margin-top: 0.5%;
|
||||
left: 0.208%;
|
||||
/*position: absolute;
|
||||
top: 0%;*/
|
||||
height: 100%;
|
||||
vertical-align: middle;
|
||||
line-height: calc(1vh + 1vh + 1.85vh);
|
||||
padding-left: 0.5vh;
|
||||
text-transform: uppercase;
|
||||
font-weight: bold;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
textarea {
|
||||
font-size: 1.65vh;
|
||||
line-height: 1.85vh;
|
||||
display: block;
|
||||
box-sizing: border-box;
|
||||
padding: 1%;
|
||||
padding-left: 3.5%;
|
||||
box-sizing: content-box;
|
||||
padding: 1vh;
|
||||
padding-left: 0.5vh;
|
||||
color: white;
|
||||
background-color: rgba(44, 62, 80, 1.0);
|
||||
width: 100%;
|
||||
border-width: 0;
|
||||
height: 3.15%;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
flex: 1;
|
||||
background-color: transparent;
|
||||
}
|
||||
|
||||
textarea:focus, input:focus {
|
||||
@@ -123,5 +152,9 @@ textarea:focus, input:focus {
|
||||
}
|
||||
|
||||
.hidden {
|
||||
display: none;
|
||||
opacity: 0;
|
||||
}
|
||||
|
||||
.hidden.animated {
|
||||
transition: opacity 1s;
|
||||
}
|
||||
4
resources/[gameplay]/chat/html/index.d.ts
vendored
Normal file
4
resources/[gameplay]/chat/html/index.d.ts
vendored
Normal file
@@ -0,0 +1,4 @@
|
||||
declare module '*.vue' {
|
||||
import Vue from 'vue'
|
||||
export default Vue
|
||||
}
|
||||
@@ -3,115 +3,12 @@
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title></title>
|
||||
<link href="vendor/latofonts.css" rel="stylesheet">
|
||||
<link href="vendor/flexboxgrid.6.3.1.min.css" rel="stylesheet"></link>
|
||||
<link href="vendor/animate.3.5.2.min.css" rel="stylesheet"></link>
|
||||
<link href="/html/vendor/latofonts.css" rel="stylesheet">
|
||||
<link href="/html/vendor/flexboxgrid.6.3.1.min.css" rel="stylesheet"></link>
|
||||
<link href="/html/vendor/animate.3.5.2.min.css" rel="stylesheet"></link>
|
||||
<link href="index.css" rel="stylesheet"></link>
|
||||
|
||||
<script src="vendor/vue.2.3.3.min.js" type="text/javascript"></script>
|
||||
<script src="config.default.js" type="text/javascript"></script>
|
||||
<script src="config.js" type="text/javascript"></script>
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
|
||||
<!-- App Template -->
|
||||
<script type="text/x-template" id="app_template">
|
||||
<div id="app">
|
||||
<div class="chat-window" :style="this.style" :class="{ 'fadeOut animated': !showWindow, 'hidden': shouldHide }">
|
||||
<div class="chat-messages" ref="messages">
|
||||
<message v-for="msg in messages"
|
||||
:templates="templates"
|
||||
:multiline="msg.multiline"
|
||||
:args="msg.args"
|
||||
:color="msg.color"
|
||||
:template="msg.template"
|
||||
:template-id="msg.templateId"
|
||||
:key="msg">
|
||||
</message>
|
||||
</div>
|
||||
</div>
|
||||
<div class="chat-input" v-show="showInput">
|
||||
<div>
|
||||
<span class="prefix">➤</span>
|
||||
<textarea v-model="message"
|
||||
ref="input"
|
||||
type="text"
|
||||
autofocus
|
||||
spellcheck="false"
|
||||
@keyup.esc="hideInput"
|
||||
@keyup="keyUp"
|
||||
@keydown="keyDown"
|
||||
@keypress.enter.prevent="send">
|
||||
</textarea>
|
||||
</div>
|
||||
<suggestions :message="message" :suggestions="suggestions">
|
||||
</suggestions>
|
||||
</div>
|
||||
</div>
|
||||
</script>
|
||||
|
||||
<!-- Message Template -->
|
||||
<script type="text/x-template" id="message_template">
|
||||
<div class="msg" :class="{ multiline }">
|
||||
<span v-html="textEscaped"></span>
|
||||
</div>
|
||||
</script>
|
||||
|
||||
<!-- Suggestions Template -->
|
||||
<script type="text/x-template" id="suggestions_template">
|
||||
<div class="suggestions-wrap" v-show="currentSuggestions.length > 0">
|
||||
<ul class="suggestions">
|
||||
<li class="suggestion" v-for="s in currentSuggestions">
|
||||
<p>
|
||||
<span :class="{ 'disabled': s.disabled }">
|
||||
{{s.name}}
|
||||
</span>
|
||||
<span class="param"
|
||||
v-for="(p, index) in s.params"
|
||||
:class="{ 'disabled': p.disabled }">
|
||||
[{{p.name}}]
|
||||
</span>
|
||||
</p>
|
||||
<small class="help">
|
||||
<template v-if="!s.disabled">
|
||||
{{s.help}}
|
||||
</template>
|
||||
<template v-for="p in s.params" v-if="!p.disabled">
|
||||
{{p.help}}
|
||||
</template>
|
||||
</small>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
</script>
|
||||
|
||||
<!-- Scripts -->
|
||||
<script type="text/javascript" src="./Suggestions.js"></script>
|
||||
<script type="text/javascript" src="./Message.js"></script>
|
||||
<script type="text/javascript" src="./App.js"></script>
|
||||
|
||||
<!-- Main Entry -->
|
||||
<script type="text/javascript">
|
||||
window.post = (url, data) => {
|
||||
var request = new XMLHttpRequest();
|
||||
request.open('POST', url, true);
|
||||
request.setRequestHeader('Content-Type', 'application/json; charset=UTF-8');
|
||||
request.send(data);
|
||||
}
|
||||
|
||||
const instance = new Vue({
|
||||
el: '#app',
|
||||
render: h => h(APP),
|
||||
});
|
||||
|
||||
window.emulate = (type, detail = {}) => {
|
||||
detail.type = type;
|
||||
window.dispatchEvent(new CustomEvent('message', {
|
||||
detail,
|
||||
}));
|
||||
};
|
||||
</script>
|
||||
|
||||
</body>
|
||||
</html>
|
||||
|
||||
7
resources/[gameplay]/chat/html/main.ts
Normal file
7
resources/[gameplay]/chat/html/main.ts
Normal file
@@ -0,0 +1,7 @@
|
||||
import Vue from 'vue';
|
||||
import App from './App.vue';
|
||||
|
||||
const instance = new Vue({
|
||||
el: '#app',
|
||||
render: h => h(App),
|
||||
});
|
||||
18
resources/[gameplay]/chat/html/tsconfig.json
Normal file
18
resources/[gameplay]/chat/html/tsconfig.json
Normal file
@@ -0,0 +1,18 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"outDir": "./",
|
||||
"module": "es6",
|
||||
"strict": true,
|
||||
"moduleResolution": "node",
|
||||
"target": "es6",
|
||||
"allowJs": true,
|
||||
"lib": [
|
||||
"es2017",
|
||||
"dom"
|
||||
]
|
||||
},
|
||||
"include": [
|
||||
"./**/*"
|
||||
],
|
||||
"exclude": []
|
||||
}
|
||||
31
resources/[gameplay]/chat/html/utils.ts
Normal file
31
resources/[gameplay]/chat/html/utils.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
export function post(url: string, data: any) {
|
||||
var request = new XMLHttpRequest();
|
||||
request.open('POST', url, true);
|
||||
request.setRequestHeader('Content-Type', 'application/json; charset=UTF-8');
|
||||
request.send(data);
|
||||
}
|
||||
|
||||
function emulate(type: string, detail = {}) {
|
||||
const detailRef = {
|
||||
type,
|
||||
...detail
|
||||
};
|
||||
|
||||
window.dispatchEvent(new CustomEvent('message', {
|
||||
detail: detailRef
|
||||
}));
|
||||
}
|
||||
|
||||
(window as any)['emulate'] = emulate;
|
||||
|
||||
(window as any)['demo'] = () => {
|
||||
emulate('ON_MESSAGE', {
|
||||
message: {
|
||||
args: [ 'me', 'hello!' ]
|
||||
}
|
||||
})
|
||||
|
||||
emulate('ON_SCREEN_STATE_CHANGE', {
|
||||
shouldHide: false
|
||||
});
|
||||
};
|
||||
File diff suppressed because one or more lines are too long
Reference in New Issue
Block a user