mirror of
https://github.com/ksyasuda/SubMiner.git
synced 2026-04-09 04:19:27 -07:00
81 lines
2.8 KiB
TypeScript
81 lines
2.8 KiB
TypeScript
export const YOMITAN_POPUP_IFRAME_SELECTOR = 'iframe.yomitan-popup, iframe[id^="yomitan-popup"]';
|
|
export const YOMITAN_POPUP_HOST_SELECTOR = '[data-subminer-yomitan-popup-host="true"]';
|
|
export const YOMITAN_POPUP_VISIBLE_HOST_SELECTOR =
|
|
'[data-subminer-yomitan-popup-host="true"][data-subminer-yomitan-popup-visible="true"]';
|
|
const YOMITAN_POPUP_VISIBLE_ATTRIBUTE = 'data-subminer-yomitan-popup-visible';
|
|
export const YOMITAN_POPUP_SHOWN_EVENT = 'yomitan-popup-shown';
|
|
export const YOMITAN_POPUP_HIDDEN_EVENT = 'yomitan-popup-hidden';
|
|
export const YOMITAN_POPUP_MOUSE_ENTER_EVENT = 'yomitan-popup-mouse-enter';
|
|
export const YOMITAN_POPUP_MOUSE_LEAVE_EVENT = 'yomitan-popup-mouse-leave';
|
|
export const YOMITAN_POPUP_COMMAND_EVENT = 'subminer-yomitan-popup-command';
|
|
export const YOMITAN_LOOKUP_EVENT = 'subminer-yomitan-lookup';
|
|
|
|
export function registerYomitanLookupListener(
|
|
target: EventTarget = window,
|
|
listener: () => void,
|
|
): () => void {
|
|
const wrapped = (): void => {
|
|
listener();
|
|
};
|
|
target.addEventListener(YOMITAN_LOOKUP_EVENT, wrapped);
|
|
return () => {
|
|
target.removeEventListener(YOMITAN_LOOKUP_EVENT, wrapped);
|
|
};
|
|
}
|
|
|
|
export function isYomitanPopupIframe(element: Element | null): boolean {
|
|
if (!element) return false;
|
|
if (element.tagName.toUpperCase() !== 'IFRAME') return false;
|
|
|
|
const hasModernPopupClass = element.classList?.contains('yomitan-popup') ?? false;
|
|
const hasLegacyPopupId = (element.id ?? '').startsWith('yomitan-popup');
|
|
return hasModernPopupClass || hasLegacyPopupId;
|
|
}
|
|
|
|
export function hasYomitanPopupIframe(root: ParentNode = document): boolean {
|
|
return (
|
|
root.querySelector(YOMITAN_POPUP_IFRAME_SELECTOR) !== null ||
|
|
root.querySelector(YOMITAN_POPUP_HOST_SELECTOR) !== null
|
|
);
|
|
}
|
|
|
|
function isVisiblePopupElement(element: Element): boolean {
|
|
const rect = element.getBoundingClientRect();
|
|
if (rect.width <= 0 || rect.height <= 0) {
|
|
return false;
|
|
}
|
|
|
|
const styles = window.getComputedStyle(element);
|
|
if (styles.visibility === 'hidden' || styles.display === 'none' || styles.opacity === '0') {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function isMarkedVisiblePopupHost(element: Element): boolean {
|
|
return element.getAttribute(YOMITAN_POPUP_VISIBLE_ATTRIBUTE) === 'true';
|
|
}
|
|
|
|
export function isYomitanPopupVisible(root: ParentNode = document): boolean {
|
|
const visiblePopupHosts = root.querySelectorAll<HTMLElement>(YOMITAN_POPUP_VISIBLE_HOST_SELECTOR);
|
|
if (visiblePopupHosts.length > 0) {
|
|
return true;
|
|
}
|
|
|
|
const popupIframes = root.querySelectorAll<HTMLIFrameElement>(YOMITAN_POPUP_IFRAME_SELECTOR);
|
|
for (const iframe of popupIframes) {
|
|
if (isVisiblePopupElement(iframe)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
const popupHosts = root.querySelectorAll<HTMLElement>(YOMITAN_POPUP_HOST_SELECTOR);
|
|
for (const host of popupHosts) {
|
|
if (isMarkedVisiblePopupHost(host)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|