// ==UserScript==
// @name Com'back
// @namespace https://greasyfork.org/fr/scripts/17200-com-back
// @version 2.1.12
// @description Ajoute un choix pour le type (écrit, audio, vidéo...) de message envoyé.
// @author DarKobalt, Naugriim(♥), Solon, Harlinde
// @match https://www.dreadcast.net/Main
// @match https://www.dreadcast.eu/Main
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_deleteValue
// @grant GM_listValues
// @license http://creativecommons.org/licenses/by-nc-nd/4.0/
// ==/UserScript==
//Un jour, j'aurai le courage pour un vrai refactoring. ;_;
//**********************************************
// PARAMETRAGE (EDITION A VOTRE PROPRE RISQUE, UTILISEZ PLUTÔT LE MENU D'OPTIONS!)
//**********************************************
//(A) Valeurs par défaut
//======================
//Dictionnaire des paramètres
let params = {};
//Icônes disponibles
params.icons = {
CLIP: "📎",
ACKN: "📨",
UPLD: "📤",
DWLD: "🔃",
FILE: "📄",
PLAY: "▶️",
WRIT: "📝",
AUDI: "🔊",
VIDE: "🎥",
DECK: "💻",
NORP: "✖",
ININ: "【",
INOU: "】",
};
//Textes informatifs.
params.infotexts = {
CLIP: "Pièce jointe",
ACKN: "Accusé de réception",
UPLD: "Envoi de données en cours",
DWLD: "Chargement en cours, veuillez patienter",
FILE: "Fichier",
PLAY: "Lecture",
WRIT: "Message écrit",
AUDI: "Message audio",
VIDE: "Message vidéo",
DECK: "Depuis un deck",
NORP: "Message HRP",
};
//Boutons disponibles :
params.actionList = ['CLIP', 'ACKN', 'UPLD', 'DWLD', 'FILE', 'PLAY'];
//Items de la liste déroulante :
params.typeList = ['WRIT', 'AUDI', 'VIDE', 'DECK', 'NORP'];
//Valeur fixée pour le menu déroulant
params.list_defaultID = 'NONE';
//Choix entre valeur fixée ou dernière valeur pour le menu déroulant (true = valeur fixée, false = dernière valeur choisie)
params.b_alwaysDefault = false;
//Dernière valeur choisie
params.list_lastID = params.list_defaultID;
//Nombre de boutons par face de carrousel
params.carousel_facesize = 6;
//Nombre d'items "utilisateur"
params.user_typeList = [];
params.user_itemsNumber = 5; //maximum
for (let i = 0; i < params.user_itemsNumber; i++) {
let user_item = "list_userItem_" + i.toString();
params.user_typeList.push(user_item);
}
//Nombre de boutons "utilisateur"
params.user_actionList = [];
params.user_actionsNumber = 6; //maximum
for (let i = 0; i < params.user_actionsNumber; i++) {
let user_action = "list_userAction_" + i.toString();
params.user_actionList.push(user_action);
}
//Couleur du texte placé entre "*".
params.emoteColor = "#58dcf9";
//(B) Sauvegarde des paramètres par défaut
//========================================
let default_params = $.extend(true, {}, params); //copie profonde
//(C) Récupération locale de paramètres
//=================================
//Valeur fixée pour le menu déroulant
if (GM_getValue("list_defaultID") !== undefined) {
params.list_defaultID = GM_getValue("list_defaultID");
}
//Choix entre valeur fixée ou dernière valeur pour le menu déroulant
if (GM_getValue("b_alwaysDefault") !== undefined) {
params.b_alwaysDefault = GM_getValue("b_alwaysDefault");
}
//Dernière valeur choisie dans le menu déroulant
if (GM_getValue("list_lastID") !== undefined) {
params.list_lastID = GM_getValue("list_lastID");
}
//Items "utilisateur"
for (let i = 0; i < params.user_itemsNumber; i++) {
let user_item = params.user_typeList[i];
if (GM_getValue(user_item) !== undefined) { //l'user_item existe en mémoire
let user_item_icon = user_item + '_icon';
let user_item_text = user_item + '_text';
if (GM_getValue(user_item_icon) !== undefined) { //récupération de l'icône
params.icons[user_item] = GM_getValue(user_item_icon);
}
if (GM_getValue(user_item_text) !== undefined) { //récupération du texte
params.infotexts[user_item] = GM_getValue(user_item_text);
}
//Ajout aux types de messages
params.typeList.push(user_item);
}
}
//Boutons "utilisateur"
for (let i = 0; i < params.user_actionsNumber; i++) {
let user_action = params.user_actionList[i];
if (GM_getValue(user_action) !== undefined) { //l'user_action existe en mémoire
let user_action_icon = user_action + '_icon';
let user_action_text = user_action + '_text';
if (GM_getValue(user_action_icon) !== undefined) { //récupération de l'icône
params.icons[user_action] = GM_getValue(user_action_icon);
}
if (GM_getValue(user_action_text) !== undefined) { //récupération du texte
params.infotexts[user_action] = GM_getValue(user_action_text);
}
//Ajout aux types d'actions
params.actionList.push(user_action);
}
}
//Nombre de faces de carrousel à créer
params.carousel_facesnumber = carouselFacesNumber();
//Couleur du texte placé entre "*".
if (GM_getValue("emoteColor") !== undefined) {
params.emoteColor = GM_getValue("emoteColor");
}
//**********************************************
//FIN PARAMETRAGE
//**********************************************
//**********************************************
//INTERFACE DE CONFIGURATION UTILISATEUR
//**********************************************
let $databox = $('#zone_dataBox');
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
//Constructeur de fenêtre de configuration
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
let DCCB_ConfigurationWindow = function () {
let window_width = '560px';
let window_height = '450px';
let $config_window = $('<div id="dccb_configwindow" onclick="engine.switchDataBox(this)"/>');
$config_window.draggable();
$config_window.addClass('dataBox focused ui-draggable');
$config_window.css({
width: window_width,
"margin-left": '-185px',
display: 'block',
position: 'absolute',
"z-index": '2',
});
for (let i = 1; i <= 8; i++) {
$('<div class="dbfond' + i + '" />').appendTo($config_window);
}
let $config_head = $('<div class="head ui-draggable-handle" ondblclick="$(\'#dccb_configwindow\').toggleClass(\'reduced\');" />').appendTo($config_window);
$('<div title="Fermer la fenêtre (Q)" class="info1 link close" onclick="engine.closeDataBox($(this).parent().parent().attr(\'id\'));" >X</div>').appendTo($config_head);
$('<div title="Reduire/Agrandir la fenêtre" class="info1 link reduce" onclick="$(\'#dccb_configwindow\').toggleClass(\'reduced\');" >-</div>').appendTo($config_head);
$('<div class="title">Configuration DC Com\'Back</div>').appendTo($config_head);
$('<div class="dbloader" />').appendTo($config_window);
let $config_content = $('<div class="content" style="height:' + window_height + '; overflow: auto"/>').appendTo($config_window);
//----------------------------------------
//Widgets internes
//----------------------------------------
let $config_interface = $('<div />').appendTo($config_content);
$config_interface.css({
"margin-left": '3px',
"font-variant": 'small-caps',
color: '#fff',
height: '100%',
width: '98%',
});
//----------------------------------------
//Bandeau incitant à fermer les messages ouverts
//----------------------------------------
let $warning_banner = $('<div />').appendTo($config_interface);
$warning_banner.text("ATTENTION : Veillez à ne pas modifier les paramètres de Com'Back si des messages sont actuellement ouverts.");
$warning_banner.css({
color: '#f10a18',
"text-align": 'center',
magin: '20px 0',
border: '1px solid #f10a18'
});
//----------------------------------------
//Configuration de l'interpréation du contenu des messages
//----------------------------------------
let $bodyconfig = $('<div />').appendTo($config_interface);
let $bodyconfig_title = $('<h2 class="couleur4" />').appendTo($bodyconfig);
$bodyconfig_title.text('Corps de message');
$bodyconfig_title.css({
"margin-bottom": '5px',
"border-bottom": '1px solid',
display: 'block',
"font-size": '17px',
"-webkit-margin-before": '0.83em',
"-webkit-margin-after": '0.83em',
"-webkit-margin-start": '0px',
"-webkit-margin-end": '0px',
"font-weight": 'bold',
position: 'relative',
});
let $bodyconfig_emote_color = $('<div class="ligne"/>').appendTo($bodyconfig);
$bodyconfig_emote_color.text('Couleur d\'emote (texte entre *...*) : ');
$bodyconfig_emote_color.css({
display: 'inline-block',
});
let $bodyconfig_emote_color_picker = $('<input type="color" />').appendTo($bodyconfig_emote_color);
$bodyconfig_emote_color_picker.val(params.emoteColor);
$bodyconfig_emote_color_picker.css({
border: '0px',
width: "150px",
});
$bodyconfig_emote_color_picker.on('input', function () {
let color = $(this).val();
params.emoteColor = color;
GM_setValue("emoteColor", color);
});
//----------------------------------------
//Configuration du menu déroulant
//----------------------------------------
let $listconfig_title = $('<h2 class="couleur4" />').appendTo($config_interface);
$listconfig_title.text('Types de message personalisés (menu déroulant)');
$listconfig_title.css({
"margin-bottom": '5px',
"border-bottom": '1px solid',
display: 'block',
"font-size": '17px',
"-webkit-margin-before": '0.83em',
"-webkit-margin-after": '0.83em',
"-webkit-margin-start": '0px',
"-webkit-margin-end": '0px',
"font-weight": 'bold',
});
//Choix de la valeur par défaut à l'ouverture d'un message ou d'un fil de discussion
let $listconfig_default = $('<div class="ligne"/>').appendTo($config_interface);
$listconfig_default.text('Valeur par défaut : ');
$listconfig_default.css({
display: 'inline-block',
"margin-bottom": '15px',
});
//Bouton-radio du choix "Dernière valeur utilisée"
let $listconfig_default_lastone = $('<input type="radio" name="typeListDefault" value="false">Dernière utilisée</input>').appendTo($listconfig_default);
$listconfig_default_lastone.css({
margin: '0 5px',
});
$listconfig_default_lastone.attr('checked', !params.b_alwaysDefault);
//Bouton-radio du choix "Valeur par défaut fixée"
let $listconfig_default_value = $('<input type="radio" name="typeListDefault" value="true">Toujours :</input>').appendTo($listconfig_default);
$listconfig_default_value.css({
margin: '0px 5px 0 25px',
"padding-left": '20px',
});
$listconfig_default_value.attr('checked', params.b_alwaysDefault);
//Menu déroulant activé si besoin d'un valeur par défaut fixée
let $listconfig_default_picker = $('<select />').appendTo($listconfig_default);
if (params.b_alwaysDefault) {
$listconfig_default_picker.removeAttr('disabled');
} else {
$listconfig_default_picker.attr('disabled', 'disabled');
}
$listconfig_default_picker.css({
"background-color": '#FFFFFF',
"-webkit-box-shadow": '0 0 1px 0px #329bc2',
"border-color": '#207695',
"border-style": 'solid',
"border-width": 'thin',
width: '175px',
margin: '0 5px',
"white-space": 'nowrap',
overflow: 'hidden',
"text-overflow": 'ellipsis',
"-o-text-overflow": 'ellipsis',
"-ms-text-overflow": 'ellipsis',
"-web-text-overflow": 'ellipsis',
"font-family": "Arial,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
});
//Gestion du choix d'une valeur dans le menu déroulant
$listconfig_default_picker.change(function () {
//Changement de la valeur par défaut de la liste déroulante
params.list_defaultID = $(this).val();
//Tentative de sauvegarde locale
GM_setValue("list_defaultID", params.list_defaultID);
});
//Ajout d'un item vide
$listconfig_default_picker.append('<option value="NONE"></option>');
//Ajout des items contenus dans typeList
for (let i = 0; i < params.typeList.length; i++) {
let type_id = params.typeList[i];
let $option = $('<option id="opt_' + type_id + '" />').appendTo($listconfig_default_picker);
let item_title = params.icons[type_id] + ' - ' + params.infotexts[type_id];
$option.val(type_id).html(item_title);
}
//Sélection du type actuellement par défaut
let type_index = params.list_defaultID;
$listconfig_default_picker.val(type_index);
//Gestion du clic sur le bouton-radio "Toujours"
$listconfig_default_value.change(function () {
if ($(this).attr('checked')) { //Utilisation de la valeur par défaut
params.b_alwaysDefault = true;
$listconfig_default_picker.removeAttr('disabled');
GM_setValue("b_alwaysDefault", params.b_alwaysDefault);
}
});
//Gestion du clic sur le bouton-radio "Dernière utilisée"
$listconfig_default_lastone.change(function () {
if ($(this).attr('checked')) { //Utilisateur de la dernière valeur utilisée
params.b_alwaysDefault = false;
params.list_defaultID = 'NONE';
$listconfig_default_picker.attr('disabled', 'disabled');
GM_setValue("list_defaultID", params.list_defaultID);
GM_setValue("b_alwaysDefault", params.b_alwaysDefault);
}
});
//Gestion des items "utilisateur"
let $listconfig_items = $('<div class="ligne"/>').appendTo($config_interface);
$listconfig_items.text('Items personnalisés : ');
let $useritems_table = $('<table id="dccb_userItems_config"/>').appendTo($listconfig_items);
$useritems_table.css({
width: '100%',
border: 'solid 1px white',
margin: '5px 0',
"font-size": '15px',
});
//Ligne d'en-têtes
$useritems_table.append($('<thead><tr><th>Symbole</th><th>Description</th><th></th></tr></thead>'));
let $useritems_tbody = $('<tbody />').appendTo($useritems_table);
for (let i = 0; i < params.user_typeList.length; i++) {
let type_id = params.user_typeList[i];
let $row = $('<tr />').appendTo($useritems_tbody);
$row.addClass("loaded_item");
$row.attr('id', type_id);
let item_icon = params.icons[type_id] || "";
let $icon_td = $('<td class="editable" style="text-align:center;width:10%;font-size: 20px;">' + item_icon + '</td>').appendTo($row);
$icon_td.data('target_type', 'icon');
let item_text = params.infotexts[type_id] || "";
let $text_td = $('<td class="editable" style="padding-left:10px;width:70%;">' + item_text + '</td>').appendTo($row);
$text_td.data('target_type', 'infotext');
//Ajout d'un bouton pour la supression
let $last_td = $('<td style="width:20%"/>').appendTo($row);
let $itemdel_btn = $('<div class="btnTxt" />').appendTo($last_td);
$itemdel_btn.data('type_ID', type_id);
$itemdel_btn.text('Supprimer');
$itemdel_btn.css({
height: '15px',
margin: '5px 15px',
});
//Handler clic sur le bouton "Supprimer" d'une ligne du tableau
$itemdel_btn.click(function () {
if ($(this).data('confirmed')) {
//Suppression des valeurs de la ligne
let type_id = $(this).data('type_ID');
$('#' + type_id + ' > td.editable').text("");
//Suppression des données "utilisateur"
//Suppresion en RAM
let index = params.typeList.indexOf(type_id);
if (index !== -1) {
params.typeList.splice(index, 1);
}
delete params.icons[type_id];
delete params.infotexts[type_id];
//Suppresion en mémoire locale
GM_deleteValue(type_id);
GM_deleteValue(type_id + '_icon');
GM_deleteValue(type_id + '_text');
//Suppression dans le menu déroulant de la fenêtre de configuration
if ($listconfig_default_picker.val() === type_id) { //l'item à supprimer est sélectionné
//On sélectionne le type NONE d'office
$listconfig_default_picker.val('NONE').trigger('change');
}
$('option#opt_' + type_id).remove();
//Remise à zéro du bouton
$(this).text('Supprimer');
$(this).data('confirmed', false);
} else {
//Besoin d'un second clic, pour confirmation
$(this).text('Confirmer');
$(this).data('confirmed', true);
}
});
$itemdel_btn.mouseleave(function () {
//Annulation de la confirmation de suppression
$(this).text('Supprimer');
$(this).data('confirmed', false);
});
}
//Css des éléments du tableau
$useritems_table.find('td').css({
border: '1px solid white',
height: '15px'
});
//Handler double-clic sur une cellule éditable
$('td.editable', $useritems_table).dblclick(function () {
let type_id = $(this).parent().attr('id');
let target_type = $(this).data('target_type');
editCellContent($(this), function (changes) {
if (changes) {
//Sauvegarde en ram et en mémoire locale
if (target_type === 'icon') {
params.icons[type_id] = changes;
GM_setValue(type_id + '_icon', changes);
} else if (target_type === 'infotext') {
params.infotexts[type_id] = changes;
GM_setValue(type_id + '_text', changes);
}
//Ajout à la liste des types disponibles (si non déjà présent)
let item_title = params.icons[type_id] + ' - ' + params.infotexts[type_id];
if (params.typeList.indexOf(type_id) === -1) {
//Ajout à la liste des types disponibles
params.typeList.push(type_id);
//Ajout d'un flag en mémoire locale
GM_setValue(type_id, true);
//Ajout au menu déroulant de la fenêtre de configuration
let $option = $('<option id="opt_' + type_id + '"/>').appendTo($listconfig_default_picker);
$option.val(type_id).html(item_title);
} else { //modification dans le menu déroulant si déjà présent
let $option = $('option#opt_' + type_id);
$option.val(type_id).html(item_title);
}
}
});
});
//----------------------------------------
//Configuration des boutons disponibles
//----------------------------------------
let $buttonsconfig_title = $('<h2 class="couleur4" />').appendTo($config_interface);
$buttonsconfig_title.text('Types d\'indications personalisés (boutons)');
$buttonsconfig_title.css({
"margin-bottom": '5px',
"border-bottom": '1px solid',
display: 'block',
"font-size": '17px',
"-webkit-margin-before": '0.83em',
"-webkit-margin-after": '0.83em',
"-webkit-margin-start": '0px',
"-webkit-margin-end": '0px',
"font-weight": 'bold',
});
//Gestion des boutons "utilisateur"
let $listconfig_actions = $('<div class="ligne"/>').appendTo($config_interface);
$listconfig_actions.text('Boutons personnalisés : ');
let $useractions_table = $('<table id="dccb_userActions_config"/>').appendTo($listconfig_actions);
$useractions_table.css({
width: '100%',
border: 'solid 1px white',
margin: '5px 0',
"font-size": '15px',
});
//Ligne d'en-têtes
$useractions_table.append($('<thead><tr><th>Symbole</th><th>Description</th><th></th></tr></thead>'));
let $useractions_tbody = $('<tbody />').appendTo($useractions_table);
for (let i = 0; i < params.user_actionList.length; i++) {
let type_id = params.user_actionList[i];
let $row = $('<tr />').appendTo($useractions_tbody);
$row.addClass("loaded_action");
$row.attr('id', type_id);
let action_icon = params.icons[type_id] || "";
let $icon_td = $('<td class="editable" style="text-align:center;width:10%;font-size: 20px;">' + action_icon + '</td>').appendTo($row);
$icon_td.data('target_type', 'icon');
let action_text = params.infotexts[type_id] || "";
let $text_td = $('<td class="editable" style="padding-left:10px;width:70%;">' + action_text + '</td>').appendTo($row);
$text_td.data('target_type', 'infotext');
//Ajout d'un bouton pour la supression
let $last_td = $('<td style="width:20%"/>').appendTo($row);
let $actiondel_btn = $('<div class="btnTxt" />').appendTo($last_td);
$actiondel_btn.data('type_ID', type_id);
$actiondel_btn.text('Supprimer');
$actiondel_btn.css({
height: '15px',
margin: '5px 15px',
});
//Handler clic sur le bouton "Supprimer" d'une ligne du tableau
$actiondel_btn.click(function () {
if ($(this).data('confirmed')) {
//Suppression des valeurs de la ligne
let type_id = $(this).data('type_ID');
$('#' + type_id + ' > td.editable').text("");
//Suppression des données "utilisateur"
//Suppresion en RAM
let index = params.actionList.indexOf(type_id);
if (index !== -1) {
params.actionList.splice(index, 1);
}
params.carousel_facesnumber = carouselFacesNumber();
delete params.icons[type_id];
delete params.infotexts[type_id];
//Suppresion en mémoire locale
GM_deleteValue(type_id);
GM_deleteValue(type_id + '_icon');
GM_deleteValue(type_id + '_text');
//Remise à zéro du bouton
$(this).text('Supprimer');
$(this).data('confirmed', false);
} else {
//Besoin d'un second clic, pour confirmation
$(this).text('Confirmer');
$(this).data('confirmed', true);
}
});
$actiondel_btn.mouseleave(function () {
//Annulation de la confirmation de suppression
$(this).text('Supprimer');
$(this).data('confirmed', false);
});
}
//Css des éléments du tableau
$useractions_table.find('td').css({
border: '1px solid white',
height: '15px'
});
//Handler double-clic sur une cellule éditable
$('td.editable', $useractions_table).dblclick(function () {
let type_id = $(this).parent().attr('id');
let target_type = $(this).data('target_type');
editCellContent($(this), function (changes) {
if (changes) {
//Sauvegarde en ram et en mémoire locale
if (target_type === 'icon') {
params.icons[type_id] = changes;
GM_setValue(type_id + '_icon', changes);
} else if (target_type === 'infotext') {
params.infotexts[type_id] = changes;
GM_setValue(type_id + '_text', changes);
}
//Ajout à la liste des actions disponibles (si non déjà présent)
if (params.actionList.indexOf(type_id) === -1) {
//Ajout à la liste des types disponibles
params.actionList.push(type_id);
params.carousel_facesnumber = carouselFacesNumber();
//Ajout d'un flag en mémoire locale
GM_setValue(type_id, true);
}
}
});
});
//----------------------------------------
//Bouton de remise à zéro des paramètres
//----------------------------------------
let $buttons_div = $('<div />').appendTo($config_interface);
$buttons_div.css({
//position: 'absolute',
width: '100%',
bottom: '0px',
});
let $reinit_btn = $('<div class="btnTxt" />').appendTo($buttons_div);
$reinit_btn.text('Remettre à zéro');
$reinit_btn.attr('title','Réinitialisation des variables et fermeture de la fenêtre');
$reinit_btn.click(function(){
//Ecrasement des paramètres par les paramètres par défaut
params = $.extend(true, {}, default_params);
//Fermeture forcée de la fenêtre de configuration
engine.closeDataBox('dccb_configwindow');
//Suppression des variables enregistrées en mémoire
let stored_values = GM_listValues();
for (let i=0;i<stored_values.length;i++){
GM_deleteValue(stored_values[i]);
}
});
this.$window = $config_window;
};
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
//FIN Constructeur de fenêtre de configuration
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
//---------------------------------------------------
//Ajout d'un item au menu bandeau "Paramètres" de DC
//---------------------------------------------------
let $params_menu = $('.menus > .parametres > ul');
let $dccb_config = $('<li />').appendTo($params_menu);
$dccb_config.text("Configuration Com'Back");
$dccb_config.addClass('link couleur2 separator');
$dccb_config.click(function () {
//Fermeture des autres instances de paramétrage ouvertes
engine.closeDataBox('dccb_configwindow');
let $config_window = new DCCB_ConfigurationWindow();
$databox.append($config_window.$window);
});
//**********************************************
// FIN INTERFACE DE CONFIGURATION UTILISATEUR
//**********************************************
//**********************************************
// FONCTIONS UTILITAIRES
//**********************************************
//Calcul du nombre de faces de carrousel
function carouselFacesNumber() {
let integer_part = Math.floor(params.actionList.length / params.carousel_facesize);
let modulo_part = params.actionList.length % params.carousel_facesize;
if (modulo_part > 0) {
return integer_part + 1;
}
return integer_part;
}
//Calcul de l'index de rattachement d'un bouton au carousel
function carouselDedicatedFaceIdx(btn_idx) {
return Math.floor(btn_idx / params.carousel_facesize);
}
//Obtention de la longueur d'un texte en pixels
function getTextWidth(text, font) {
// re-use canvas object for better performance
let canvas = getTextWidth.canvas || (getTextWidth.canvas = document.createElement("canvas"));
let context = canvas.getContext("2d");
context.font = font;
let metrics = context.measureText(text);
return metrics.width;
}
//Mise en forme |n * \n|[icône + texte]|n * \n|
function makeBanner(CR_before, id, CR_after, isHeader) {
let banner = params.icons.ININ + ' ' + params.icons[id] + ' - ' + params.infotexts[id] + ' ' + params.icons.INOU;
if (!isHeader) return Array(CR_before + 1).join("\n") + banner + Array(CR_after + 1).join("\n"); //Si ce n'est pas un en-tête, on ne se préoccupe pas de centrer le texte et on applique directement les retours à la ligne.
let bannerWidth = Math.round(getTextWidth(banner, "12px Trebuchet MS")); //Largeur de la bannière
let spaceWidth = Math.round(getTextWidth(" ", "12px Trebuchet MS")); //Largeur d'un espace insécable (alt+255, différent de l'espace normal)
let windowsWidth = 300; //Largeur de la fenêtre à l'endroit de l'en-tête dans laquelle on peut écrire (avec avatar et marges soustraits)
let nbrSpace = Math.floor(((windowsWidth - bannerWidth) / 2) / spaceWidth); //Déduction du nombre d'espaces à ajouter
let space = " ";
return Array(CR_before + 1).join("\n") + space.repeat(nbrSpace) + banner + Array(CR_after + 1).join("\n");
}
//Troncature de chaînes de caractères
function truncateString(string, nb_char) {
return $.trim(string).substring(0, nb_char).split(" ").slice(0, -1).join(" ") + "...";
}
//Vérification des données, actuellement désactivé
function checkData() {
/*if (params.list_defaultID === undefined) return false;
if (params.list_lastID === undefined) return false;
if (params.b_alwaysDefault === undefined) return false;
if (params.icons === undefined) return false;
if (params.infotexts === undefined) return false;
for (let i = 0; i < params.actionList.length; i++) {
if (params.icons[params.actionList[i]] === undefined) return false;
if (params.infotexts[params.actionList[i]] === undefined) return false;
}
for (let i = 0; i < params.typeList.length; i++) {
if (params.icons[params.typeList[i]] === undefined) return false;
if (params.infotexts[params.typeList[i]] === undefined) return false;
}*/
return true;
}
//Edition d'une cellule de tableau
function editCellContent(cell, cb) {
let init_value = cell.text();
cell.html('<input style="width:100%;background-color:rgb(200,200,210)" type="text" value="' + init_value + '" />');
cell.children().first().focus();
cell.children().first().keypress(function (e) {
if (e.which == 13) { //Touche entrée appuyée
let new_value = cell.find('input').val();
cell.text(new_value);
if (new_value !== init_value) { //la nouvelle valeur est différente de l'ancienne
return cb(new_value);
} else { //pas de changement de valeur
return cb(false);
}
}
});
//Le champ d'édition n'a plus le focus = un clic a été donné sur un autre élément
cell.children().first().blur(function () {
cell.text(init_value);
return cb(false);
});
}
//Formatage des liens et des contenus de message
function format_liens(html) {
//URLs starting with http://, https://, or ftp://
let replacePattern1 = /(\b(https?|ftp):\/\/[-A-Z0-9+&#\/%?=~_|!:,.;]*[-A-Z0-9+&#\/%=~_|])/gim;
html = html.replace(replacePattern1, '<a href="$1" target="_blank">$1</a>');
//URLs starting with "www." (without // before it, or it'd re-link the ones done above).
let replacePattern2 = /(^|[^\/])(www\.[\S]+(\b|$))/gim;
html = html.replace(replacePattern2, '$1<a href="http://$2" target="_blank">$2</a>');
html = html.replace(/(<br\/><\/a>)|(<br><\/a>)/gim, '<\/a><br>'); //Problème des liens www dont la fin peut être tronquée avec une balise <br/>
html = html.replace(/(<br\/>\" target)|(<br>\" target)/gim, '" target'); //Pareil, correction dans le href
/*HARLINDE COURTESY*/
//Ta mère youtube
html = html.replace(/<a\shref=\"(?:http?s?:\/\/)?(?:www\.)?(?:youtube\.com|youtu\.be)\/(?:watch\?v=)?(.+)\"\starget=\"_blank\">[\S]+<\/a>/gim, '<center><embed style="max-width: 355px;" src="https://www.youtube.com/embed/$1"></embed></center>');
//Transforme les liens de son en son...
html = html.replace(/<a\shref=\"([\S]+(\.mp3|\.ogg|\.wav))\"\starget=\"_blank\">[\S]+<\/a>/gim, '<center><audio controls><source src="$1"></audio></center>');
//Transforme les liens d'images en images cliquables
html = html.replace(/<a\shref=\"([\S]+(\.png|\.jpg|\.jpeg|\.gif))\"\starget=\"_blank\">[\S]+<\/a>/gim, '<center><a href="$1" target="_blank"><img src="$1" style="max-width: 355px;"><\/a></center>');
//Tranforme le texte entre * en italique
html = html.replace(/\*([^\*]+)\*/gim, '<span style="font-style: italic; color: ' + params.emoteColor + ';">$1</span>');
return html;
}
//**********************************************
// FIN FONCTIONS UTILITAIRES
//**********************************************
//**********************************************
// FONCTIONS DES INTERFACES MESSAGES
//**********************************************
//Pour un nouveau message
function mainf() {
var old_id = "#db_new_message";
var $databox = $(old_id);
var new_id = 'db_message_' + new Date().getTime().toString();
$databox.attr('id', new_id);
var db_id = '#'+new_id;
var class_name = ".message_nouveau";
var toContent = db_id + " > relative > div.content";
var $msg_content = $(toContent);
var $msg_textarea = $(toContent + " > " + class_name + " > #nm_texte > textarea");
//Edition du bouton pour réduire la fenêtre afin de corriger le onclick en chemin relatif jQuery
//Ainsi que du double clic sur le titre de la fenêtre qui a le même effet
$(db_id + " > relative > .head").attr("ondblclick", "").dblclick(function(){
$(this).parent().toggleClass('reduced');
});
$(db_id + " > relative > .head > .info1.link.reduce").attr("onclick", "").click(function(){
$(this).parent().parent().toggleClass('reduced');
});
//Menu déroulant
//*********************
//Création du conteneur
var $types_div = $('<div id="DCCB_divListe" />').appendTo($msg_content);
$types_div.css({
"z-index": '999999',
position: 'absolute',
top: '3px',
left: '37px',
"background-color": '#ACABAB',
});
//Création de la liste
var $types_selection = $('<select id="listeTypes" />').appendTo($types_div);
$types_selection.css({ //TODO : vérifier que ellipsis fonctionne
display: 'block',
width: '165px',
"white-space": 'nowrap',
overflow: 'hidden',
"text-overflow": 'ellipsis',
"-o-text-overflow": 'ellipsis',
"-ms-text-overflow": 'ellipsis',
"-web-text-overflow": 'ellipsis',
"font-family": "Arial,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
});
$types_selection.change(function () {
var type_id = $(this).val();
//Changement de dernier item choisi
params.list_lastID = type_id;
//Ajout d'une infobulle
if (type_id !== 'NONE') {
$(this).attr('title', makeBanner(0, type_id, 0, false));
} else {
$(this).attr('title', "");
}
//Sauvegarde locale
GM_setValue("list_lastID", type_id);
});
//Ajout d'un élément neutre
$types_selection.append('<option value="NONE"></option>');
//Ajout des éléments en fonction de 'typeList'
for (var i = 0; i < params.typeList.length; i++) {
var $option = $('<option />').appendTo($types_selection);
var type_id = params.typeList[i];
var item_title = params.icons[type_id] + ' - ' + params.infotexts[type_id];
$option.val(type_id).html(item_title);
}
//Application du choix par défaut
var type_id = (params.b_alwaysDefault) ? params.list_defaultID : params.list_lastID;
$types_selection.val(type_id);
if (type_id !== 'NONE') {
$types_selection.attr('title', makeBanner(0, type_id, 0, false));
} else {
$types_selection.attr('title', "");
}
//Boutons
//*********************
//Edit bouton d'envoi pour injecter fonctions customs
$(db_id + " > relative > .content > .message_nouveau > .envoyer.link").attr("onclick", "").click(function() {
//Ajout d'un en-tête au message avant l'envoi
if ($types_selection.val() !== 'NONE') {
var header = makeBanner(1, $types_selection.val(), 4, true);
var new_msg = header + $msg_textarea.val();
$msg_textarea.val(new_msg);
}
nav.getMessagerie().sendMessage($(db_id));
$(this).off("click"); //Empêche un envoi multiple du message.
});
//Boutons annexes pour ajouter des bouts de texte (pièce jointe, etc).
var $actions_div = $('<div id="div_cb_annexes"/>').appendTo($msg_content);
$actions_div.css({
"z-index": '999999',
position: 'absolute',
top: '25%',
right: '4px',
width: '30px',
height: (params.carousel_facesize * 30).toString() + 'px',
overflow: 'hidden',
border: '1px solid rgba(0, 0, 0,0.1)',
"box-shadow": '0',
});
$actions_div.on('contextmenu', function (e) {
e.stopPropagation();
e.preventDefault();
if (carousel_stockpile.length < 2) {
return false; // Ignorer si pas plus d'éléments
}
//On extrait le premier élément de la pile (sans remise)
var fifo = carousel_stockpile.shift();
//On cache cet élément
$('.' + fifo, $actions_div).hide();
//On montre le nouvel élément de tête
$('.' + carousel_stockpile[0], $actions_div).show();
//On ajoute l'ancien premier élément en fin de pile
carousel_stockpile.push(fifo);
return false;
});
//Initialisation d'une pile de gestion des faces du carrousel
var carousel_stockpile = []
for (var idx_carousel = 0; idx_carousel < params.carousel_facesnumber; idx_carousel++) {
carousel_stockpile.push('carousel_' + idx_carousel.toString());
}
//Création des boutons rattachés à une face du carrousel
for (var idx_btn = 0; idx_btn < params.actionList.length; idx_btn++) {
var dedicatedCarousel_id = "carousel_" + carouselDedicatedFaceIdx(idx_btn);
var action_id = params.actionList[idx_btn];
var $button = $('<button title="' + params.infotexts[action_id] + '" class="cb_annexes" id="DCCB_b' + idx_btn.toString() + '">' + params.icons[action_id] + '</button>').appendTo($actions_div);
//On range le bouton sur une face du carrousel via une classe CSS
$button.addClass(dedicatedCarousel_id);
$button.val(action_id);
//On cache tous les boutons à l'initialisation
$button.hide();
$button.click(function () {
var innerBanner = makeBanner(0, $(this).val(), 1, false);
var new_msg = $msg_textarea.val() + innerBanner;
$msg_textarea.val(new_msg);
});
}
//On montre les membres de la première face du carrousel
$('.carousel_0', $actions_div).show();
$(".cb_annexes").css({
"background-color": "#ACABAB",
"height": "30px",
"width": "30px",
"font-size": "20px",
"font-family": "Arial,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
}); //Ajout du CSS des boutons.
//console.log("Com'back started: nouveau message"); //Debug
}
//Pour un film de discussion existant
function filcomf() {
$("#liste_messages").ajaxComplete(function (e, xhr, opt) { //Naugriim, je t'aime. <3 (Attendre le chargement de la fenêtre avant d'envoyer la sauce)
//console.log("COM'BACK DEBUG: " + opt.url);
if(!opt.url.includes("Menu/Messaging/action=OpenMessage")) return;
$("#liste_messages").unbind('ajaxComplete'); //Evite de renvoyer à chaque nouvelle requête ajax du jeu et donc de dupliquer la fonction
let message_id = $("input.id_conversation").attr('value'); //Récupère l'id du message
let db_id = "#db_message_" + message_id;
$(db_id + " > relative > .content > .message > .grid1 > .contenu").css("min-height", "25vh"); // Test pour hauteur de com' responsive...
//Transformation des liens en liens cliquables
//*********************
let $message_content = $(db_id + " > relative > .content > .message > .grid1 > .contenu > .texte");
let content_orig = $message_content.html();
$message_content.html(format_liens(content_orig));
let last_clicked_id = $(db_id + " .link.conversation.selected").attr('id');
$(db_id + " .link.conversation").click(function () {
let $cheminTexte = $(db_id + " > relative > .content > .message > .grid1 > .contenu > .texte");
$cheminTexte.ajaxComplete(function () {
$(this).unbind('ajaxComplete');
let this_clicked_id = $(db_id + " .link.conversation.selected").attr('id');
if (this_clicked_id !== last_clicked_id) {
last_clicked_id = this_clicked_id; //Eviter de repasser la fonction qui sinon nique les liens.
$(this).html(format_liens($(this).html()));
}
});
});
$(db_id + " > relative > .content > div.message > .grid2 .btnTxt").filter((i, el) => ["Répondre", "Inviter"].includes($(el).text())).click(function () { //Création et affichage lors du clic sur l'un des boutons en bas de la fenêtre
if (document.getElementById("dccb_div_fc_" + message_id) === null) { //Ne recrée pas l'interface du script si elle existe déjà: REND IMPOSSIBLE L'OUVERTURE DE PLUSIEURS COMS SANS BUGS.
let $msg_content = $(db_id + " > relative > .content > .message");
let $msg_textarea = $(db_id + " > relative > .content > .message > .zone_reponse > .texte > #nm_texte > textarea");
//Augmentation de la taille de la zone_conversation
//$('.zone_conversation').css('height', '340px');
//Version animée
/* $('.zone_conversation').animate({
height: '340px'
}, 500);*/
//Menu déroulant
//*********************
//Création du conteneur
let $types_div = $('<div id="dccb_div_fc_' + message_id + '" />').appendTo($msg_content);
$types_div.addClass('dccb_div_fc');
$types_div.css({
"z-index": '999999',
position: 'absolute',
bottom: '0px',
left: '105px',
"background-color": '#FFFFFF',
"-webkit-box-shadow": '0 0 1px 0px #329bc2',
"border-color": '#207695',
"border-style": 'solid',
"border-width": 'thin',
});
//Création de la liste
let $types_selection = $('<select id="listeTypesfc_' + message_id + '" />').appendTo($types_div);
$types_selection.addClass('listeTypesfc');
$types_selection.css({ //TODO : vérifier que ellipsis fonctionne
height: '27px',
color: '#397d94',
display: 'block',
width: '250px',
"white-space": 'nowrap',
overflow: 'hidden',
"text-overflow": 'ellipsis',
"-o-text-overflow": 'ellipsis',
"-ms-text-overflow": 'ellipsis',
"-web-text-overflow": 'ellipsis',
"font-family": 'Arial,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif',
});
$types_selection.change(function () {
let type_id = $(this).val();
//Changement de dernier item choisi
params.list_lastID = type_id;
//Ajout d'une infobulle
if (type_id !== 'NONE') {
$(this).attr('title', makeBanner(0, type_id, 0, false));
} else {
$(this).attr('title', "");
}
//Sauvegarde locale
GM_setValue("list_lastID", type_id);
});
//Ajout d'un élément neutre
$types_selection.append('<option value="NONE"></option>');
//Ajout des éléments en fonction de 'typeList'
for (let i = 0; i < params.typeList.length; i++) {
let $option = $('<option />').appendTo($types_selection);
let type_id = params.typeList[i];
let item_title = params.icons[type_id] + ' - ' + params.infotexts[type_id];
$option.val(type_id).html(item_title);
}
//Application du choix par défaut
let type_id = (params.b_alwaysDefault) ? params.list_defaultID : params.list_lastID;
$types_selection.val(type_id);
if (type_id !== 'NONE') {
$types_selection.attr('title', makeBanner(0, type_id, 0, false));
} else {
$types_selection.attr('title', "");
}
//Boutons
//*********************
//Bouton d'envoi injecté
$(db_id + " > relative > .content > .message > .zone_reponse > .btnTxt").attr("onclick", "").click(function(){
//Ajout d'un en-tête au message avant l'envoi
if ($types_selection.val() !== 'NONE') {
let header = makeBanner(1, $types_selection.val(), 4, true);
let new_msg = header + $msg_textarea.val();
$msg_textarea.val(new_msg);
}
nav.getMessagerie().sendMessage($(db_id));
$(this).off("click"); //Empêche d'envoyer un message en double.
});
//Boutons annexes pour ajouter des bouts de texte (pièce jointe, etc).
let $actions_div = $('<div id="dccb_annexesfc_' + message_id + '"/>').appendTo($msg_content);
$actions_div.css({
"z-index": '999999',
position: 'absolute',
bottom: '-1px',
left: '365px',
height: '29px',
width: (params.carousel_facesize * 30).toString() + 'px',
overflow: 'hidden',
border: '1px solid rgba(0, 0, 0,0.1)',
"box-shadow": '0',
});
$actions_div.on('contextmenu', function (e) {
e.stopPropagation();
e.preventDefault();
if (carousel_stockpile.length < 2) {
return false; // Ignorer si pas plus d'éléments
}
//On extrait le premier élément de la pile (sans remise)
let fifo = carousel_stockpile.shift();
//On cache cet élément
$('.' + fifo, $actions_div).hide();
//On montre le nouvel élément de tête
$('.' + carousel_stockpile[0], $actions_div).show();
//On ajoute l'ancien premier élément en fin de pile
carousel_stockpile.push(fifo);
return false;
});
//Initialisation d'une pile de gestion des faces du carrousel
let carousel_stockpile = []
for (let idx_carousel = 0; idx_carousel < params.carousel_facesnumber; idx_carousel++) {
carousel_stockpile.push('carousel_' + idx_carousel.toString());
}
for (let idx_btn = 0; idx_btn < params.actionList.length; idx_btn++) {
let dedicatedCarousel_id = "carousel_" + carouselDedicatedFaceIdx(idx_btn);
let action_id = params.actionList[idx_btn];
let $button = $('<button title="' + params.infotexts[action_id] + '" class="cb_annexesfc cb_annexesfc_' + message_id + '" id="DCCB_b' + idx_btn.toString() + '_' + message_id + '">' + params.icons[action_id] + '</button>').appendTo($actions_div);
$button.val(action_id);
$button.addClass(dedicatedCarousel_id);
//On cache tous les boutons à l'initialisation
$button.hide();
$button.click(function () {
let innerBanner = makeBanner(0, $(this).val(), 1, false);
let new_msg = $msg_textarea.val() + innerBanner;
$msg_textarea.val(new_msg);
});
}
$('.carousel_0', $actions_div).show();
$(".cb_annexesfc_" + message_id).css({
"color": "#397D94",
"background-color": "#FFFFFF",
"height": "29px",
"width": "29px",
"font-size": "15px",
"border-color": "#207695",
"font-family": "Arial,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
}); //Ajout du CSS des boutons.
$msg_textarea.css({
"font-family": "Verdana,Courier,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
}); //CSS Unicode pour la zone d'écriture d'un fil de com'. Nécessite d'être placé ici parce que nique AJAX et jQuery.
}
});
});
console.log("Com'back started: fil de discussion"); //Debug
}
//**********************************************
// FIN FONCTIONS DES INTERFACES MESSAGES
//**********************************************
//**********************************************
// MAIN
//**********************************************
if (!checkData()) {
console.log("DCCB - Com'back : Erreur dans les données");
} else {
$("#zone_messagerie > div.btnTxt.link").click(mainf); //Nouveaux messages
//$("#liste_contacts div.btnTxt.mail").click(mainf); //Nouveaux messages (Liste de contacts)
$("li.message").click(filcomf); //1ère initialisation
$(document).ajaxSuccess(function(e, xhr, opt){
if (opt.url.includes("/Menu/Messaging/OpenFolder")) {
//console.log("DCCB - Actualisation des events");
$("li.message").off("click", filcomf);
$("li.message").click(filcomf);
}
});
setInterval(function () { //Fix bien crade pour contacter l'auteur d'une annonce AITL.
$(".annonce > .texte > span:contains(Contacter l'auteur)").click(mainf);
}, 1000);
//Affichage du Unicode via la police Unifont
$("body").css({
"font-family": "Trebuchet MS,Verdana,Arial,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
});
$("textarea").css({
"font-family": "Verdana,Courier,Segoe UI Symbol,Unifont,Unifont Upper CSUR,sans-serif",
}); //Ne fonctionne que pour les nouveaux messages à cause d'AJAX. Sera sûrement à adapter si Remedy règle le bug de fermeture des nouveaux messages.
console.log("DCCB - Com'back initialisé!");
}