Adds video search buttons to anime lists on MAL.
// ==UserScript==
// @name MAL VS
// @version 1.0
// @description Adds video search buttons to anime lists on MAL.
// @author Ivan Yancharkin
// @license MIT
// @namespace http://github.com/yancharkin/malvs
// @homepageURL https://github.com/yancharkin/malvs
// @supportURL https://github.com/yancharkin/malvs/issues
// @icon https://github.com/yancharkin/malvs/blob/webextension/assets/icons/icon-48.png?raw=true
// @match *://myanimelist.net/animelist/*
// @match *://myanimelist.net/anime/*
// @grant GM.setValue
// @grant GM.getValue
// ==/UserScript==
(async () => {
let icon = '';
let sitesList = await GM.getValue('sitesList', "https://www.crunchyroll.com/search?from=&q=");
sitesList = sitesList.split(",")
function removeButtons() {
let buttons = document.getElementsByClassName("searchanimebutton");
while (buttons.length > 0) {
buttons[0].parentNode.removeChild(buttons[0]);
};
};
function setupButton(button, title){
button.src = icon;
button.alt = title;
button.style.width = "32px";
button.style.cursor = "pointer";
button.onmouseover = function(){
this.style.filter="brightness(1.5)";
this.style.transform="scale(-1.1, 1.1)";
};
button.onmouseout = function(){
this.style.filter="brightness(1)";
this.style.transform="scale(1, 1)";
};
button.onclick = function(){loadDataAndSearch(this.alt)};
return button;
};
function addButtons() {
let allAnimes = document.getElementsByClassName("data title clearfix");
let existingButtons = document.getElementsByClassName("searchanimebutton");
if (existingButtons.length < allAnimes.length) {
for (let anime of allAnimes) {
if (anime.parentElement.getElementsByClassName("searchanimebutton").length === 0) {
let title = anime.firstChild.innerHTML;
let button = document.createElement("img");
button = setupButton(button, title);
let malvsTd = document.createElement("td");
malvsTd.className = "data searchanimebutton";
malvsTd.appendChild(button);
anime.parentElement.appendChild(malvsTd);
}
}
}
};
function loadDataAndSearch(title) {
for (let site of sitesList) {
if (!site.startsWith("-")) {
window.open(site + title);
}
};
};
function addSettingsButton() {
async function settingsEdit() {
let newList = prompt("Sites list: ", await GM.getValue('sitesList', "https://www.crunchyroll.com/search?from=&q="));
if (newList) {
await GM.setValue('sitesList', newList);
newList = newList.split(",")
sitesList = newList;
};
};
let settingsDiv = document.createElement("div");
settingsDiv.className = "icon-menu";
let settingsText = document.createElement("span");
settingsText.className = "text";
settingsText.innerHTML = "MAL VS";
settingsText.style.cursor = "pointer";
settingsText.onclick = settingsEdit;
let settingsButton = document.createElement("img");
settingsButton.src = icon;
settingsButton.style.transform="scaleX(-1)";
settingsButton.alt = "MAL VS Settings";
settingsButton.style.width = "24px";
settingsButton.style.height = "24px";
settingsButton.style.position = "absolute";
settingsButton.style.left = "13px";
settingsButton.style.top = "13px";
settingsButton.style.cursor = "pointer";
settingsButton.onmouseover = function() {this.style.filter="brightness(1.5)"};
settingsButton.onmouseout = function() {this.style.filter="brightness(1)"};
settingsButton.onclick = settingsEdit;
settingsDiv.appendChild(settingsButton);
settingsDiv.appendChild(settingsText);
let listMenu = document.getElementsByClassName("list-menu-float")[0];
let settingsMenu;
if (!listMenu) {
settingsDiv.className = "icon-menu";
let menuPlaceholder = document.createElement("div");
menuPlaceholder.className = "list-menu-float";
menuPlaceholder.appendChild(settingsDiv);
document.body.appendChild(menuPlaceholder);
settingsMenu = menuPlaceholder;
} else {
listMenu.appendChild(settingsDiv);
settingsMenu = listMenu;
}
let listMenuOffset = Math.round((window.innerHeight - settingsMenu.offsetHeight)/2);
if (listMenuOffset < 0) listMenuOffset = 0;
settingsMenu.style.top = listMenuOffset + "px";
};
const regexp = RegExp("myanimelist.net/animelist//*")
let curUrl = window.location.href;
if (regexp.test(curUrl)) {
addSettingsButton();
if (document.getElementsByClassName("malvs-header").length === 0) {
let tableHeader = document.getElementsByClassName("list-table-header")[0]
let malvsHeader = document.createElement("th");
malvsHeader.className = "header-title malvs-header";
malvsHeader.innerHTML ='<a class="link hover_info">MAL VS</a>';
tableHeader.appendChild(malvsHeader);
};
removeButtons();
setInterval(addButtons, 1000);
} else {
for (let malvsButton of document.getElementsByClassName("searchanimebutton")) {
malvsButton.parentNode.removeChild(malvsButton);
}
let titleDiv = document.getElementsByClassName("h1-title")[0]
titleDiv.style.verticalAlign = "middle";
titleDiv.style.float = "none";
let targetElement = document.getElementsByClassName("h1 edit-info")[0];
targetElement.style.minHeight = "32px";
let title;
try {
title = document.getElementsByClassName("title-name h1_bold_none")[0].firstElementChild.textContent;
} catch(error) {
console.log(error);
title = document.getElementsByClassName("title-name")[0].innerHTML;
}
let malvsDiv = document.createElement("div");
malvsDiv.className = "searchanimebutton";
malvsDiv.style.display = "table-cell";
malvsDiv.style.verticalAlign = "middle";
malvsDiv.style.width = "48px";
malvsDiv.style.transform = "scaleX(-1)";
malvsDiv.style.textAlign = "right";
let button = document.createElement("img");
button = setupButton(button, title);
malvsDiv.appendChild(button);
targetElement.prepend(malvsDiv);
};
})();