Affiche les avatars améliorés des PJs avec couleurs de faction, bordures stylisées, infos profil (fonction + domiciliation) en ligne. Version stable visuellement harmonieuse (v4.4.0). 🦝
// ==UserScript==
// @name Kraland - Avatars personnalisés
// @namespace http://tampermonkey.net/
// @version 4.4.0
// @description Affiche les avatars améliorés des PJs avec couleurs de faction, bordures stylisées, infos profil (fonction + domiciliation) en ligne. Version stable visuellement harmonieuse (v4.4.0). 🦝
// @author Racket Raccoon / Th3rd
// @match http://www.kraland.org/*
// @match https://www.kraland.org/*
// @icon 
// @license MIT
// @run-at document-start
// ==/UserScript==
(function () {
'use strict';
// Définition des constantes visuelles de base (taille des avatars, opacité, tailles de police)
const AVATAR_SIZE = '75px';
const PNJ_SIZE = '50px';
const FONT_SMALL = '0.85em';
const OPACITY_LIGHT = '0.9';
// Couleurs associées à chaque drapeau de nation
const flagColors = {
'f1.png': '#B22222', 'f2.png': '#8B5A2B', 'f3.png': '#DAA520', 'f4.png': '#055cb5',
'f5.png': '#025e19', 'f6.png': '#7A1C81', 'f7.png': '#262424', 'f8.png': '#008080', 'f9.png': '#556B2F'
};
// Cache pour éviter de refaire des requêtes inutiles aux profils
const functionCache = {};
// Icônes Lucide
const lucideIcons = {
briefcase: '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="width:14px;height:14px;margin-right:4px;vertical-align:middle;opacity:0.7"><rect x="2" y="7" width="20" height="14" rx="2" ry="2"/><path d="M16 3h-8v4h8V3z"/></svg>',
mapPin: '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="width:14px;height:14px;margin-right:4px;vertical-align:middle;opacity:0.7"><path d="M12 21s6-5.686 6-10a6 6 0 1 0 -12 0c0 4.314 6 10 6 10z"/><circle cx="12" cy="11" r="2"/></svg>',
user: '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="width:14px;height:14px;margin-right:4px;vertical-align:middle;opacity:0.7"><path d="M20 21v-2a4 4 0 0 0 -4-4H8a4 4 0 0 0 -4 4v2"/><circle cx="12" cy="7" r="4"/></svg>',
link: '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="width:14px;height:14px;margin-right:4px;vertical-align:middle;opacity:0.7"><path d="M10 13a5 5 0 0 1 7 0l1.5 1.5a5 5 0 0 1 0 7 5 5 0 0 1 -7 0L10 20a5 5 0 0 1 0-7m-6-6a5 5 0 0 1 7 0l1.5 1.5a5 5 0 0 1 0 7 5 5 0 0 1 -7 0L4 10a5 5 0 0 1 0-7"/></svg>'
};
// Applique une typographie à la cellule de nom
function styleNameCell(nameCell) {
nameCell.style.fontFamily = "'Roboto', sans-serif";
nameCell.style.lineHeight = '1.6';
}
// Crée une <div> contenant une info stylisée (fonction ou domiciliation)
function createInfoDiv(content, options = {}) {
const div = document.createElement('div');
div.innerHTML = content;
div.style.fontSize = FONT_SMALL;
div.style.opacity = OPACITY_LIGHT;
if (options.color) div.style.color = options.color;
if (options.italic) div.style.fontStyle = 'italic';
if (options.bold) div.style.fontWeight = 'bold';
if (options.marginTop) div.style.marginTop = options.marginTop;
if (options.className) div.className = options.className;
return div;
}
// Insère sous le nom du PJ sa fonction et éventuellement sa domiciliation
function insertFunctionBelowName(row, functionText, domiciliationText = '', placeholder = false) {
const nameCell = row.querySelectorAll('td')[1];
if (nameCell?.querySelector('.avatar-function')) return;
styleNameCell(nameCell);
const isLoading = placeholder && functionText === 'Chargement...';
const isEmpty = !functionText || functionText === '-' || (!placeholder && functionText === 'Chargement...');
const funcDiv = createInfoDiv(
isLoading ? 'Chargement...' : (isEmpty ? lucideIcons.briefcase + 'Aucune fonction' : lucideIcons.briefcase + functionText),
{
color: isEmpty ? '#666' : (isLoading ? '#999' : ''),
italic: isEmpty || isLoading,
bold: !(isEmpty || isLoading),
marginTop: '0.1em',
className: 'avatar-function'
}
);
nameCell.appendChild(funcDiv);
if (domiciliationText) {
const domDiv = createInfoDiv(lucideIcons.mapPin + domiciliationText, { italic: true });
nameCell.appendChild(domDiv);
}
const nameLink = nameCell.querySelector('a');
if (nameLink) {
nameLink.style.fontWeight = 'bold';
nameLink.style.fontSize = '1.1em';
}
}
// Applique une couleur de fond (semi-transparente) à toute la ligne du PJ selon sa nationnalité
function colorRow(row, color) {
row.querySelectorAll('td').forEach(cell => {
cell.style.setProperty('background-color', `${color}4D`, 'important');
cell.style.setProperty('background-clip', 'padding-box', 'important');
});
// Cible aussi directement la cellule contenant l’avatar s’il y a un lien avec l’id
const avatarCell = row.querySelector('td.tdbc');
if (avatarCell) {
avatarCell.style.setProperty('background-color', `${color}4D`, 'important');
}
}
// Crée un conteneur autour de l'avatar avec style et bordure
function createAvatarWrapper(img, color, linkHref) {
const isPNJ = img.src.includes('/npc/');
const wrapper = document.createElement('a');
wrapper.href = linkHref || '#';
wrapper.style.display = 'inline-block';
wrapper.style.position = 'relative';
wrapper.style.width = isPNJ ? PNJ_SIZE : AVATAR_SIZE;
wrapper.style.height = isPNJ ? PNJ_SIZE : AVATAR_SIZE;
wrapper.style.backgroundColor = 'transparent';
wrapper.style.zIndex = '1';
if (isPNJ) {
wrapper.style.border = `3px solid ${color}`;
wrapper.style.borderRadius = '0';
} else {
wrapper.style.borderRadius = '50%';
}
img.style.width = isPNJ ? 'auto' : AVATAR_SIZE;
img.style.height = isPNJ ? 'auto' : AVATAR_SIZE;
img.style.maxHeight = isPNJ ? PNJ_SIZE : '';
img.style.maxWidth = isPNJ ? 'none' : '';
img.style.zIndex = '2';
let imgContainer;
if (isPNJ) {
wrapper.style.border = `3px solid ${color}`;
wrapper.style.borderRadius = '0';
imgContainer = document.createElement('div');
imgContainer.style.width = PNJ_SIZE;
imgContainer.style.height = PNJ_SIZE;
imgContainer.style.overflow = 'hidden';
imgContainer.style.display = 'flex';
imgContainer.style.alignItems = 'center';
imgContainer.style.justifyContent = 'center';
imgContainer.style.position = 'relative';
img.style.position = 'relative';
img.style.width = '100%';
img.style.height = '100%';
img.style.objectFit = 'cover';
img.style.transform = 'none';
imgContainer.appendChild(img);
} else {
img.style.borderRadius = '50%';
img.style.position = 'relative';
imgContainer = img;
const overlay = document.createElement('div');
overlay.style.position = 'absolute';
overlay.style.inset = '0';
overlay.style.borderRadius = '50%';
overlay.style.boxShadow = `inset 0 0 0 4px ${color}`;
overlay.style.pointerEvents = 'none';
overlay.style.zIndex = '3';
wrapper.appendChild(overlay);
}
wrapper.appendChild(imgContainer);
return wrapper;
}
// Traitement des avatars pour les personnages joueurs (PJs)
function handlePJAvatar(img, row, td) {
const flagImg = Array.from(row.querySelectorAll('img')).find(img => {
const file = img.src.split('/').pop();
return flagColors.hasOwnProperty(file);
});
const color = flagImg ? flagColors[flagImg.src.split('/').pop()] : '#888';
const link = td.querySelector('a[href*="main.php?p=6_1_0_1"]');
const match = link?.href.match(/p1=(\d+)/);
const wrapper = createAvatarWrapper(img, color, link?.href);
td.replaceChildren(wrapper);
// Ajout fond sur la cellule AVATAR
td.style.setProperty('background-color', `${color}4D`, 'important');
colorRow(row, color); // garde la coloration des autres cellules
// Vérifie si le personnage est recherché (n'importe où dans la cellule de nom)
const nameCell = row.querySelectorAll('td')[1];
if (nameCell) {
const originalHTML = nameCell.innerHTML;
if (/\[recherchée?\]/i.test(originalHTML)) {
nameCell.innerHTML = originalHTML.replace(/\s*\[recherchée?\]/gi, '');
// Ajout du label "Recherché"
const label = document.createElement('div');
label.textContent = 'RECHERCHÉ';
Object.assign(label.style, {
position: 'absolute',
bottom: '2px',
left: '50%',
transform: 'translateX(-50%)',
background: 'rgba(192,57,43,0.9)',
color: '#fff',
padding: '2px 6px',
fontSize: '1em',
borderRadius: '4px',
pointerEvents: 'none',
whiteSpace: 'nowrap',
zIndex: '6',
fontFamily: "'Roboto', sans-serif",
lineHeight: '1.6'
});
wrapper.appendChild(label);
}
}
if (match) {
const observer = new IntersectionObserver(entries => {
entries.forEach(entry => {
if (entry.isIntersecting) {
fetchFunctionFromProfile(match[1], row);
observer.disconnect();
}
});
});
observer.observe(row);
}
}
// Traitement des avatars pour les PNJs (mise en forme différente)
function handlePNJAvatar(img, row) {
const td = img.closest('td.tdbc');
if (!td) return;
const flagImg = Array.from(row.querySelectorAll('img')).find(img => {
const file = img.src.split('/').pop();
return flagColors.hasOwnProperty(file);
});
const color = flagColors[flagImg?.src.split('/').pop()] || '#444';
colorRow(row, color);
td.replaceChildren(createAvatarWrapper(img, color));
const nameCell = row.querySelectorAll('td')[1];
if (nameCell) {
styleNameCell(nameCell);
const nameLink = nameCell.querySelector('a');
if (nameLink) {
nameLink.style.fontWeight = 'normal';
nameLink.style.fontSize = '1.1em';
}
let nameText = nameCell.innerHTML;
nameText = nameText.replace(/<p[^>]*>/g, '').replace(/<\/p>/g, '').replace(/<br><br>/g, '<br>');
const regex = /(.+?)\s*\[(.+?)\]/;
const match = nameText.match(regex);
if (match) {
const name = match[1].trim();
const rawTitle = match[2].trim();
const title = rawTitle.replace(/^\p{L}/u, c => c.toUpperCase()); // Majuscule Unicode
let titleIcon = title === 'Capturé' ? lucideIcons.link : lucideIcons.user;
let titleHtml = `${titleIcon}<i>${title}</i>`;
let nameHtml = `<a href="${nameLink.href}" style="font-weight: bold; font-size: 1.1em;">${name}</a>`;
if (!nameText.includes(titleHtml)) {
nameCell.innerHTML = `${nameHtml}<br>${titleHtml}`;
} else {
nameCell.innerHTML = nameHtml;
}
}
}
}
// Récupère les infos "fonction" et "domiciliation" depuis la page de profil d’un personnage
async function fetchFunctionFromProfile(persoId, row) {
if (functionCache[persoId] !== undefined) {
insertFunctionBelowName(row, functionCache[persoId].func, functionCache[persoId].dom);
return;
}
insertFunctionBelowName(row, 'Chargement...', '', true);
try {
const res = await fetch(`main.php?p=6_1_0_1&p1=${persoId}`);
if (!res.ok) throw new Error('Network response was not ok');
const buffer = await res.arrayBuffer();
const html = new TextDecoder('iso-8859-1').decode(buffer);
const doc = new DOMParser().parseFromString(html, 'text/html');
const tds = Array.from(doc.querySelectorAll('td'));
const fonction = tds.find(td => td.textContent.trim() === 'Fonction')?.nextElementSibling?.textContent.trim() || '';
const domiciliation = tds.find(td => td.textContent.trim() === 'Domiciliation')?.nextElementSibling?.textContent.trim() || '';
functionCache[persoId] = { func: fonction, dom: domiciliation };
row.querySelector('.avatar-function')?.remove();
insertFunctionBelowName(row, fonction, domiciliation);
} catch (error) {
insertFunctionBelowName(row, 'Erreur de chargement', '', true);
}
}
// Vérifie si une ligne <tr> appartient à un groupe (nécessaire pour ne pas traiter les lignes isolées)
function isInsideGroupRow(row) {
let prev = row?.previousElementSibling;
while (prev) {
if (prev.tagName === 'TR' && prev.textContent.includes('Groupe')) return true;
prev = prev.previousElementSibling;
}
return false;
}
// Traite toutes les lignes du tableau pour modifier les avatars
function resizeAvatars() {
document.querySelectorAll('table .tdbc img').forEach(img => {
const src = img.src;
if (src.includes('img.kraland.org/2/mat/')) return;
const td = img.closest('td.tdbc');
const row = img.closest('tr');
if (!row || !td || !isInsideGroupRow(row)) return;
if (row.querySelector('a[href*="order.php?p1=200"]')) return;
const isPNJ = src.includes('img.kraland.org/2/npc/');
if (isPNJ) return handlePNJAvatar(img, row);
handlePJAvatar(img, row, td);
});
}
// Ajoute un séparateur entre deux groupes affichés dans la même table
function insertGroupSeparator() {
const rows = document.querySelectorAll('table tr');
let groupCount = 0;
rows.forEach(row => {
const hdr = row.querySelector('th a[href*="order.php"]');
const isSpecial = row.querySelector('th a[href="order.php?p1=3004"]');
if (!hdr || isSpecial) return;
groupCount++;
const next = row.nextElementSibling;
if (groupCount > 1 && next && next.querySelector('td')) {
const sep = document.createElement('tr');
sep.innerHTML = '<td colspan="2"> </td>';
row.parentNode.insertBefore(sep, row);
}
});
}
// Lancement du traitement une fois que le DOM est chargé
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', () => {
resizeAvatars();
insertGroupSeparator();
});
} else {
resizeAvatars();
insertGroupSeparator();
}
// Ajout de styles CSS pour les en-têtes de groupe arrondis
const style = document.createElement('style');
style.textContent = `
.thb {
padding: 8px;
text-align: center;
border-radius: 15px;
border-left: none !important;
border-right: none !important;
border-top: none !important;
border-bottom: none !important;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
}
`;
document.head.appendChild(style);
})();