Break expression sync cycle and guard portrait lookup
This commit is contained in:
@@ -0,0 +1,73 @@
|
||||
import {
|
||||
syncedExpressionPortraits,
|
||||
getSyncedExpressionPortrait
|
||||
} from '../core/state.js';
|
||||
import {
|
||||
isSafeImageSrc,
|
||||
normalizeImageSrc,
|
||||
resolveImageUrl
|
||||
} from './imageUrls.js';
|
||||
|
||||
function normalizeName(name) {
|
||||
return String(name || '').trim().toLowerCase();
|
||||
}
|
||||
|
||||
function namesMatch(a, b) {
|
||||
const left = normalizeName(a);
|
||||
const right = normalizeName(b);
|
||||
if (!left || !right) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return left === right || left.startsWith(right + ' ') || right.startsWith(left + ' ');
|
||||
}
|
||||
|
||||
function isDocumentLikeUrl(src) {
|
||||
const candidate = resolveImageUrl(src);
|
||||
if (!candidate) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const current = new URL(window.location.href);
|
||||
return candidate.origin === current.origin
|
||||
&& candidate.pathname === current.pathname
|
||||
&& candidate.search === current.search;
|
||||
}
|
||||
|
||||
export function isUsableExpressionSrc(src) {
|
||||
const normalized = normalizeImageSrc(src);
|
||||
if (!normalized) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const lower = normalized.toLowerCase();
|
||||
if (lower.includes('/img/default-expressions/') || lower.includes('/default-expressions/')) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (isDocumentLikeUrl(normalized)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return isSafeImageSrc(normalized);
|
||||
}
|
||||
|
||||
export function getExpressionPortraitForCharacter(characterName) {
|
||||
const target = normalizeName(characterName);
|
||||
if (!target) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const exact = getSyncedExpressionPortrait(target);
|
||||
if (isUsableExpressionSrc(exact)) {
|
||||
return exact;
|
||||
}
|
||||
|
||||
for (const [storedName, src] of Object.entries(syncedExpressionPortraits)) {
|
||||
if (namesMatch(storedName, target) && isUsableExpressionSrc(src)) {
|
||||
return src;
|
||||
}
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
@@ -226,7 +226,7 @@ export function resolvePresentCharacterPortrait(name) {
|
||||
}
|
||||
}
|
||||
|
||||
if (this_chid !== undefined && characters[this_chid]?.name && namesMatch(characters[this_chid].name, name)) {
|
||||
if (this_chid !== undefined && characters?.[this_chid]?.name && namesMatch(characters[this_chid].name, name)) {
|
||||
const thumbnailUrl = getSafeThumbnailUrl('avatar', characters[this_chid].avatar);
|
||||
if (thumbnailUrl) {
|
||||
return thumbnailUrl;
|
||||
|
||||
Reference in New Issue
Block a user