rDrama/files/assets/js/emoji_modal.js

530 lines
15 KiB
JavaScript
Raw Normal View History

2022-07-16 21:00:02 +00:00
/*
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
2022-09-04 23:15:37 +00:00
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2022-07-16 21:00:02 +00:00
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
2022-09-04 23:15:37 +00:00
along with this program. If not, see <https://www.gnu.org/licenses/>.
2022-07-16 21:00:02 +00:00
Copyright (C) 2022 Dr Steven Transmisia, anti-evil engineer,
2022-09-04 23:15:37 +00:00
2022 Nekobit, king autist
2022-07-16 21:00:02 +00:00
*/
// Status
let emojiEngineStarted = false;
// DOM stuff
const classesSelectorDOM = document.getElementById("emoji-modal-tabs");
const emojiButtonTemplateDOM = document.getElementById("emoji-button-template");
const emojiResultsDOM = document.getElementById("tab-content");
const emojiSelectSuffixDOMs = document.getElementsByClassName("emoji-suffix");
const emojiSelectPostfixDOMs= document.getElementsByClassName("emoji-postfix");
const emojiNotFoundDOM = document.getElementById("no-emojis-found");
const emojiWorkingDOM = document.getElementById("emojis-work");
const emojiSearchBarDOM = document.getElementById('emoji_search');
let emojiInputTargetDOM = undefined;
// Emojis usage stats. I don't really like this format but I'll keep it for backward comp.
const favorite_emojis = JSON.parse(localStorage.getItem("favorite_emojis")) || {};
/** Associative array of all the emojis' DOM */
let emojiDOMs = {};
let globalEmojis;
2022-07-16 21:00:02 +00:00
const EMOIJ_SEARCH_ENGINE_MIN_INTERVAL = 350;
let emojiSearcher = {
working: false,
queries: [],
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
addQuery: function(query)
{
this.queries.push(query);
if(!this.working)
this.work();
},
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
work: async function work() {
this.working = true;
while(this.queries.length > 0)
{
const startTime = Date.now();
// Get last input
const query = this.queries[this.queries.length - 1].toLowerCase();
this.queries = [];
// To improve perf we avoid showing all emojis at the same time.
if(query === "")
{
await classesSelectorDOM.children[0].children[0].click();
classesSelectorDOM.children[0].children[0].classList.add("active");
continue;
}
// Search
const resultSet = emojisSearchDictionary.completeSearch(query);
2022-09-04 23:15:37 +00:00
// update stuff
2022-07-16 21:00:02 +00:00
for(const [emojiName, emojiDOM] of Object.entries(emojiDOMs))
emojiDOM.hidden = !resultSet.has(emojiName);
emojiNotFoundDOM.hidden = resultSet.size !== 0;
let sleepTime = EMOIJ_SEARCH_ENGINE_MIN_INTERVAL - (Date.now() - startTime);
if(sleepTime > 0)
await new Promise(r => setTimeout(r, sleepTime));
}
this.working = false;
}
};
// tags dictionary. KEEP IT SORT
class EmoijsDictNode
{
constructor(tag, name) {
this.tag = tag;
this.emojiNames = [name];
}
}
const emojisSearchDictionary = {
dict: [],
updateTag: function(tag, emojiName) {
if(tag === undefined || emojiName === undefined)
return;
let low = 0;
let high = this.dict.length;
while (low < high) {
let mid = (low + high) >>> 1;
if (this.dict[mid].tag < tag)
low = mid + 1;
else
high = mid;
}
let target = low;
if(this.dict[target] !== undefined && this.dict[target].tag === tag)
this.dict[target].emojiNames.push(emojiName);
else
this.dict.splice(target ,0,new EmoijsDictNode(tag, emojiName));
},
/**
* We also check for substrings! (sigh)
* @param {String} tag
* @returns {Set}
*/
completeSearch: function(query) {
const result = new Set();
for(let i = 0; i < this.dict.length; i++)
if(this.dict[i].tag.includes(query))
for(let j = 0; j < this.dict[i].emojiNames.length; j++)
result.add(this.dict[i].emojiNames[j])
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
return result;
}
};
// get public emojis list
const emojiRequest = new XMLHttpRequest();
emojiRequest.open("GET", '/emojis.csv');
emojiRequest.setRequestHeader('xhr', 'xhr');
emojiRequest.onload = async () => {
2022-09-04 23:15:37 +00:00
let emojis = JSON.parse(emojiRequest.response);
2022-07-16 21:00:02 +00:00
if(! (emojis instanceof Array ))
2022-07-17 17:05:24 +00:00
throw new TypeError("[EMOJI DIALOG] rDrama's server should have sent a JSON-coded Array!");
2022-07-16 21:00:02 +00:00
globalEmojis = emojis.map(({name, author, count}) => ({name, author, count}));
2022-07-16 21:00:02 +00:00
let classes = new Set();
const bussyDOM = document.createElement("div");
for(let i = 0; i < emojis.length; i++)
{
const emoji = emojis[i];
emojisSearchDictionary.updateTag(emoji.name, emoji.name);
if(emoji.author !== undefined && emoji.author !== null)
emojisSearchDictionary.updateTag(emoji.author.toLowerCase(), emoji.name);
if(emoji.tags instanceof Array)
for(let i = 0; i < emoji.tags.length; i++)
emojisSearchDictionary.updateTag(emoji.tags[i], emoji.name);
classes.add(emoji.kind);
2022-07-16 21:00:02 +00:00
// Create emoji DOM
const emojiDOM = document.importNode(emojiButtonTemplateDOM.content, true).children[0];
emojiDOM.title = emoji.name
if(emoji.author !== undefined && emoji.author !== null)
emojiDOM.title += "\nauthor\t" + emoji.author
if(emoji.count !== undefined)
emojiDOM.title += "\nused\t" + emoji.count;
emojiDOM.dataset.className = emoji.kind;
2022-07-16 21:00:02 +00:00
emojiDOM.dataset.emojiName = emoji.name;
emojiDOM.onclick = emojiAddToInput;
2022-09-04 23:15:37 +00:00
emojiDOM.hidden = true;
2022-07-16 21:00:02 +00:00
const emojiIMGDOM = emojiDOM.children[0];
emojiIMGDOM.src = "/e/" + emoji.name + ".webp";
emojiIMGDOM.alt = emoji.name;
/** Disableing lazy loading seems to reduce cpu usage somehow (?)
2022-09-04 23:15:37 +00:00
* idk it is difficult to benchmark */
2022-07-16 21:00:02 +00:00
emojiIMGDOM.loading = "lazy";
// Save reference
emojiDOMs[emoji.name] = emojiDOM;
// Add to the document!
bussyDOM.appendChild(emojiDOM);
}
// Create header
for(let className of classes)
{
let classSelectorDOM = document.createElement("li");
classSelectorDOM.classList.add("nav-item");
let classSelectorLinkDOM = document.createElement("a");
classSelectorLinkDOM.href = "#";
classSelectorLinkDOM.classList.add("nav-link", "emojitab");
classSelectorLinkDOM.dataset.bsToggle = "tab";
classSelectorLinkDOM.dataset.className = className;
classSelectorLinkDOM.innerText = className;
2023-01-29 09:19:04 +00:00
classSelectorLinkDOM.addEventListener('click', switchEmojiTab);
2022-07-16 21:00:02 +00:00
classSelectorDOM.appendChild(classSelectorLinkDOM);
classesSelectorDOM.appendChild(classSelectorDOM);
}
// Show favorite for start.
await classesSelectorDOM.children[0].children[0].click();
2022-09-04 23:15:37 +00:00
// Send it to the render machine!
2022-07-16 21:00:02 +00:00
emojiResultsDOM.appendChild(bussyDOM);
emojiResultsDOM.hidden = false;
emojiWorkingDOM.hidden = true;
emojiSearchBarDOM.disabled = false;
}
/**
2022-09-04 23:15:37 +00:00
*
* @param {Event} e
*/
2022-07-16 21:00:02 +00:00
function switchEmojiTab(e)
{
const className = e.currentTarget.dataset.className;
emojiSearchBarDOM.value = "";
focusSearchBar(emojiSearchBarDOM);
2022-07-16 21:00:02 +00:00
emojiNotFoundDOM.hidden = true;
// Special case: favorites
if(className === "favorite")
{
for(const emojiDOM of Object.values(emojiDOMs))
emojiDOM.hidden = true;
2023-01-03 10:46:54 +00:00
const favs = Object.keys(Object.fromEntries(
Object.entries(favorite_emojis).sort(([,a],[,b]) => b-a)
)).slice(0, 25);
for (const emoji of favs)
if(emojiDOMs[emoji] instanceof HTMLElement)
emojiDOMs[emoji].hidden = false;
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
return;
}
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
for(const emojiDOM of Object.values(emojiDOMs))
emojiDOM.hidden = emojiDOM.dataset.className !== className;
}
2022-12-30 12:14:18 +00:00
for (const emojitab of document.getElementsByClassName('emojitab')) {
emojitab.addEventListener('click', (e)=>{switchEmojiTab(e)})
}
2022-07-16 21:00:02 +00:00
async function start_search() {
emojiSearcher.addQuery(emojiSearchBarDOM.value.trim());
2022-07-16 21:00:02 +00:00
// Remove any selected tab, now it is meaningless
for(let i = 0; i < classesSelectorDOM.children.length; i++)
classesSelectorDOM.children[i].children[0].classList.remove("active");
}
/**
2022-09-04 23:15:37 +00:00
* Add the selected emoji to the targeted text area
* @param {Event} event
*/
2022-07-16 21:00:02 +00:00
function emojiAddToInput(event)
{
// This should not happen if used properly but whatever
if(!(emojiInputTargetDOM instanceof HTMLTextAreaElement) && !(emojiInputTargetDOM instanceof HTMLInputElement))
return;
2022-09-04 23:15:37 +00:00
let strToInsert = event.currentTarget.dataset.emojiName;
2022-07-16 21:00:02 +00:00
for(let i = 0; i < emojiSelectPostfixDOMs.length; i++)
if(emojiSelectPostfixDOMs[i].checked)
strToInsert = strToInsert + emojiSelectPostfixDOMs[i].value;
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
for(let i = 0; i < emojiSelectSuffixDOMs.length; i++)
if(emojiSelectSuffixDOMs[i].checked)
strToInsert = emojiSelectSuffixDOMs[i].value + strToInsert;
strToInsert = ":" + strToInsert + ":"
2023-03-18 14:27:12 +00:00
insertText(emojiInputTargetDOM, strToInsert)
2022-09-04 23:15:37 +00:00
2022-07-16 21:00:02 +00:00
// kick-start the preview
emojiInputTargetDOM.dispatchEvent(new Event('input'));
// Update favs. from old code
if (favorite_emojis[event.currentTarget.dataset.emojiName])
favorite_emojis[event.currentTarget.dataset.emojiName] += 1;
else
favorite_emojis[event.currentTarget.dataset.emojiName] = 1;
localStorage.setItem("favorite_emojis", JSON.stringify(favorite_emojis));
}
2022-09-08 17:12:46 +00:00
let emoji_typing_state = false;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
function update_ghost_div_textarea(text)
{
let ghostdiv = text.parentNode.querySelector(".ghostdiv");
if (!ghostdiv) return;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
ghostdiv.innerText = text.value.substring(0, text.selectionStart);
2022-12-07 20:27:22 +00:00
ghostdiv.insertAdjacentHTML('beforeend', "<span></span>");
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
// Now lets get coordinates
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
ghostdiv.style.display = "initial";
let end = ghostdiv.querySelector("span");
const carot_coords = end.getBoundingClientRect();
const ghostdiv_coords = ghostdiv.getBoundingClientRect();
ghostdiv.style.display = "none";
return { pos: text.selectionStart, x: carot_coords.x, y: carot_coords.y - ghostdiv_coords.y };
}
// Used for anything where a user is typing, specifically for the emoji modal
// Just leave it global, I don't care
let speed_carot_modal = document.createElement("div");
speed_carot_modal.id = "speed-carot-modal";
speed_carot_modal.style.position = "absolute";
speed_carot_modal.style.left = "0px";
speed_carot_modal.style.top = "0px";
speed_carot_modal.style.display = "none";
document.body.appendChild(speed_carot_modal);
let e
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
let current_word = "";
let selecting;
let emoji_index = 0;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
function curr_word_is_emoji()
{
return current_word && current_word.charAt(0) == ":" &&
current_word.charAt(current_word.length-1) != ":";
}
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
function populate_speed_emoji_modal(results, textbox)
{
selecting = true;
2022-09-08 17:12:46 +00:00
if (!results || results.size === 0)
2022-09-04 23:15:37 +00:00
{
2022-09-08 17:12:46 +00:00
speed_carot_modal.style.display = "none";
return -1;
2022-09-04 23:15:37 +00:00
}
2022-09-08 17:12:46 +00:00
emoji_index = 0;
speed_carot_modal.innerHTML = "";
const MAXXX = 25;
2022-09-08 17:12:46 +00:00
// Not sure why the results is a Set... but oh well
let i = 0;
for (let emoji of results)
2022-09-04 23:15:37 +00:00
{
const name = emoji.name
if (i++ > MAXXX) return i;
2022-09-08 17:12:46 +00:00
let emoji_option = document.createElement("div");
emoji_option.className = "speed-modal-option emoji-option " + (i === 1 ? "selected" : "");
emoji_option.tabIndex = 0;
let emoji_option_img = document.createElement("img");
emoji_option_img.className = "speed-modal-image emoji-option-image";
// This is a bit
emoji_option_img.src = `/e/${name}.webp`;
2022-09-08 17:12:46 +00:00
let emoji_option_text = document.createElement("span");
emoji_option_text.title = name;
if(emoji.author !== undefined && emoji.author !== null)
emoji_option_text.title += "\nauthor\t" + emoji.author
if(emoji.count !== undefined)
emoji_option_text.title += "\nused\t" + emoji.count;
emoji_option_text.innerText = name;
if (current_word.includes("#")) name = `#${name}`
if (current_word.includes("!")) name = `!${name}`
2022-09-08 17:12:46 +00:00
emoji_option.addEventListener('click', () => {
2022-09-08 17:12:46 +00:00
selecting = false;
2022-09-04 23:15:37 +00:00
speed_carot_modal.style.display = "none";
textbox.value = textbox.value.replace(new RegExp(current_word+"(?=\\s|$)", "g"), `:${name}: `)
textbox.focus()
if (document.location.pathname != '/chat'){
markdown(textbox)
}
});
2022-09-08 17:12:46 +00:00
// Pack
emoji_option.appendChild(emoji_option_img);
emoji_option.appendChild(emoji_option_text);
speed_carot_modal.appendChild(emoji_option);
2022-09-04 23:15:37 +00:00
}
2022-09-08 17:12:46 +00:00
if (i === 0) speed_carot_modal.style.display = "none";
else speed_carot_modal.style.display = "initial";
return i;
}
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
function update_speed_emoji_modal(event)
{
const box_coords = update_ghost_div_textarea(event.target);
2022-09-04 23:15:37 +00:00
let text = event.target.value;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
// Unused, but left incase anyone wants to use this more efficient method for emojos
switch (event.data)
{
case ':':
emoji_typing_state = true;
break;
case ' ':
emoji_typing_state = false;
break;
default:
break;
}
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
// Get current word at string, such as ":marse" or "word"
let coords = text.indexOf(' ',box_coords.pos);
2022-12-30 16:47:50 +00:00
current_word = /:[!#a-zA-Z0-9_]+(?=\n|$)/.exec(text.slice(0, coords === -1 ? text.length : coords));
2022-09-08 17:12:46 +00:00
if (current_word) current_word = current_word.toString();
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
/* We could also check emoji_typing_state here, which is less accurate but more efficient. I've
* kept it unless someone wants to provide an option to toggle it for performance */
if (curr_word_is_emoji() && current_word != ":")
{
loadEmojis(null);
let modal_pos = event.target.getBoundingClientRect();
modal_pos.x += window.scrollX;
modal_pos.y += window.scrollY;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
speed_carot_modal.style.display = "initial";
speed_carot_modal.style.left = box_coords.x - 35 + "px";
speed_carot_modal.style.top = modal_pos.y + box_coords.y + 14 + "px";
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
// Do the search (and do something with it)
const resultSet = emojisSearchDictionary.completeSearch(current_word.substr(1).replace(/#/g, "").replace(/!/g, ""))
const found = globalEmojis.filter(i => resultSet.has(i.name));
populate_speed_emoji_modal(found, event.target);
2022-09-04 23:15:37 +00:00
}
2022-09-08 17:12:46 +00:00
else {
speed_carot_modal.style.display = "none";
}
}
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
function speed_carot_navigate(e)
{
if (!selecting) return;
let select_items = speed_carot_modal.querySelectorAll(".speed-modal-option");
2022-11-07 10:49:49 +00:00
if (!select_items || !curr_word_is_emoji()) return;
2022-09-08 17:12:46 +00:00
// Up or down arrow or enter
if (e.keyCode == 38 || e.keyCode == 40 || e.keyCode == 13)
2022-09-04 23:15:37 +00:00
{
2022-09-08 17:12:46 +00:00
if (emoji_index > select_items.length)
emoji_index = select_items;
2022-09-08 17:12:46 +00:00
select_items[emoji_index].classList.remove("selected");
switch (e.keyCode)
2022-09-04 23:15:37 +00:00
{
2022-09-08 17:12:46 +00:00
case 38: // Up arrow
if (emoji_index)
emoji_index--;
break;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
case 40: // Down arrow
if (emoji_index < select_items.length-1) emoji_index++;
break;
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
case 13:
select_items[emoji_index].click();
2022-09-04 23:15:37 +00:00
2022-09-08 17:12:46 +00:00
default:
break;
2022-09-04 23:15:37 +00:00
}
2022-09-08 17:12:46 +00:00
select_items[emoji_index].classList.add("selected");
e.preventDefault();
}
}
2022-09-08 17:12:46 +00:00
// Let's get it running now
let forms = document.querySelectorAll("textarea, .allow-emojis");
forms.forEach(i => {
let pseudo_div = document.createElement("div");
pseudo_div.className = "ghostdiv";
pseudo_div.style.display = "none";
i.after(pseudo_div);
i.addEventListener('input', update_speed_emoji_modal, false);
i.addEventListener('keydown', speed_carot_navigate, false);
});
2022-07-16 21:00:02 +00:00
function loadEmojis(inputTargetIDName)
{
2022-12-30 16:47:50 +00:00
selecting = false;
speed_carot_modal.style.display = "none";
2022-07-16 21:00:02 +00:00
if(!emojiEngineStarted)
{
emojiEngineStarted = true;
emojiRequest.send();
}
2022-09-04 23:15:37 +00:00
if (inputTargetIDName)
emojiInputTargetDOM = document.getElementById(inputTargetIDName);
2022-07-16 21:00:02 +00:00
}
document.getElementById('emojiModal').addEventListener('shown.bs.modal', function () {
focusSearchBar(emojiSearchBarDOM);
setTimeout(() => {
focusSearchBar(emojiSearchBarDOM);
}, 200);
2022-12-19 22:18:32 +00:00
setTimeout(() => {
focusSearchBar(emojiSearchBarDOM);
2022-12-19 22:18:32 +00:00
}, 1000);
2022-11-07 10:49:49 +00:00
});