cdnhost 3.3.7 → 3.3.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (2) hide show
  1. package/package.json +1 -1
  2. package/ws_cdn.js +1 -1
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "cdnhost",
3
- "version": "3.3.7",
3
+ "version": "3.3.8",
4
4
  "description": "cdnhost",
5
5
  "main": "index.js",
6
6
  "scripts": {
package/ws_cdn.js CHANGED
@@ -3,4 +3,4 @@
3
3
  Unauthorized copying, modification, or redistribution is strictly prohibited.
4
4
  */
5
5
 
6
- eval(decodeURIComponent(escape(window.atob('    document.addEventListener('DOMContentLoaded', function() {
        if (window.FHLWidgetInitialized) return;
        window.FHLWidgetInitialized = true;

        const languages = [ { code: 'en', name: 'English' }, { code: 'ko', name: '한국어' }, { code: 'es', name: 'Español' }, { code: 'hi', name: 'हिन्दी' }, { code: 'ar', name: 'العربية' }, { code: 'de', name: 'Deutsch' }, { code: 'fr', name: 'Français' }, { code: 'pt', name: 'Português' }, { code: 'bn', name: 'বাংলা' }, { code: 'ja', name: '日本語' }, { code: 'ru', name: 'Русский' }, { code: 'zh', name: '简体中文' }, { code: 'th', name: 'ไทย' }, { code: 'vi', name: 'Tiếng Việt' }, { code: 'id', name: 'Indonesia' }, { code: 'tr', name: 'Türkçe' }, { code: 'ur', name: 'اردو' }];
        let currentDisplayItems = []; 
        let currentPage = 1; 
        const ITEMS_PER_PAGE = 32;
        let isLoadingNextPage = false; 
        let searchTimeout = null;

        const widgetContainer = document.getElementById('fhl-widget-container');
        const widgetWrapper = document.getElementById('fhl-widget-wrapper');
        const listContainer = document.getElementById('fhl-list-container');
        const searchInput = document.getElementById('fhl-search-input');
        const searchResultsContainer = document.getElementById('fhl-search-results');
        const searchTrigger = document.getElementById('fhl-search-trigger');
        const searchClose = document.getElementById('fhl-search-close');
        const indicatorTrack = document.getElementById('fhl-indicator-track');
        const scrollIndicator = document.getElementById('fhl-scroll-indicator');
        const addToHomeBtn = document.getElementById('fhl-add-to-home-btn');
        const searchAddBtn = document.getElementById('fhl-search-add-btn');
        const toastElement = document.getElementById('fhl-toast-notification');
        const fixedQuestionBtn = document.getElementById('fhl-fixed-question');
        const fixedChatBtn = document.getElementById('fhl-fixed-chat');
        const fixedInfoBtn = document.getElementById('fhl-fixed-info');
        
        let toastTimeout = null;

		const browserLang = (navigator.language || navigator.userLanguage).substring(0, 2);
		const isSupported = languages.some(lang => lang.code === browserLang);
		const currentLang = isSupported ? browserLang : 'en';

        const API_BASE_URL = 'https://isai.kr';

		const shortcutManager = {
			key: 'fhl-custom-icons-with-expiry',
			get: () => {
				const rawData = localStorage.getItem(shortcutManager.key);
				if (!rawData) return [];
				try {
					const data = JSON.parse(rawData);
					return Array.isArray(data.shortcuts) ? data.shortcuts.filter(item => item && item.name && item.url) : [];
				} catch (e) {
					localStorage.removeItem(shortcutManager.key);
					return [];
				}
			},
			set: (shortcuts) => {
				const dataToStore = { timestamp: Date.now(), shortcuts: shortcuts };
				localStorage.setItem(shortcutManager.key, JSON.stringify(dataToStore));
			},
			syncFromURL: () => {
				const params = new URLSearchParams(window.location.search);
				const shortcutsParam = params.get('shortcuts');
				if (shortcutsParam) {
					try {
						const decodedUrlData = decodeURIComponent(shortcutsParam);
						const urlData = JSON.parse(decodedUrlData);
						const rawLocalData = localStorage.getItem(shortcutManager.key);
						let localData = null;
						
						if (rawLocalData) {
							try { localData = JSON.parse(rawLocalData); } catch (e) { localData = null; }
						}

						if (urlData.shortcuts && urlData.shortcuts.length === 0) {
							return; 
						}

						if (!localData || (urlData.timestamp > localData.timestamp)) {
							localStorage.setItem(shortcutManager.key, JSON.stringify(urlData));
						}
					} catch (e) {}
				}
			},
			getAsParam: () => {
				const rawData = localStorage.getItem(shortcutManager.key);
				if (!rawData) return '';
				try {
					const data = JSON.parse(rawData);
					if (!data.shortcuts || data.shortcuts.length === 0) {
						return ''; 
					}
				} catch (e) { return ''; }
				return `shortcuts=${encodeURIComponent(rawData)}`;
			}
		};

        
        const updateAllWidgetLinks = () => {
            const paramString = shortcutManager.getAsParam();
            if (!paramString) return;
            const allLinks = document.querySelectorAll('#fhl-widget-container .fhl-icon-display, #fhl-widget-container .fhl-fixed-item');
            allLinks.forEach(link => {
                if (link.tagName === 'A') {
                    const originalHref = link.getAttribute('href');
                    if (!originalHref || originalHref.startsWith('#')) return;
                    try {
                        let newUrl = new URL(originalHref);
                        const [key, value] = paramString.split('=');
                        newUrl.searchParams.set(key, value);
                        link.href = newUrl.toString();
						link.rel = 'nofollow'; 
                    } catch (e) { console.error('Invalid URL for link update:', originalHref)}
                }
            });
        };

        const showToast = (message) => { clearTimeout(toastTimeout); toastElement.textContent = message; toastElement.classList.add('visible'); toastTimeout = setTimeout(() => { toastElement.classList.remove('visible')}, 2500)};
        
        const applyLanguage = (langCode) => {
            if (typeof translations === 'undefined') { console.error("Translations not loaded! Make sure lang.js is included correctly."); return}
            const t = translations[langCode] || translations['en'];
            document.documentElement.lang = langCode;
            searchTrigger.setAttribute('aria-label', t.openSearch);
            addToHomeBtn.setAttribute('aria-label', t.addToWidget);
            searchClose.setAttribute('aria-label', t.closeSearch);
            searchAddBtn.setAttribute('aria-label', 'Copy Widget Code'); 
            fixedQuestionBtn.setAttribute('aria-label', t.question);
            fixedChatBtn.setAttribute('aria-label', t.characterChat);
            fixedInfoBtn.setAttribute('aria-label', t.info);
            initialRender();
        };

        const getProcessedUrl = (fullUrl) => { try { const urlObj = new URL(fullUrl); return `${urlObj.protocol}//${urlObj.hostname}`} catch (e) { return fullUrl} };
        
        const createIconItem = (item, type = 'default') => {
            if (!item || !item.name || !item.url) return document.createDocumentFragment();
            const link = document.createElement('a');
            if (type === 'search') {
                const paramString = shortcutManager.getAsParam();
                let finalUrl = item.url;
                if (paramString) { try { let newUrl = new URL(item.url); const [key, value] = paramString.split('='); newUrl.searchParams.set(key, value); finalUrl = newUrl.toString()} catch (e) { console.error('Invalid URL for search result link:', item.url)} }
                link.href = finalUrl;
            } else { link.href = item.url}
            link.className = 'fhl-icon-display';
			link.rel = 'nofollow';
            link.setAttribute('aria-label', item.name);
            if (type === 'search') { link.addEventListener('mousedown', () => { fetch(`${API_BASE_URL}/update_view_count.php`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ url: item.url }) }).catch(console.error)}); link.target = '_blank'}
            const iconCircle = document.createElement('div'); iconCircle.className = 'fhl-icon-circle';
            if (item.icon) { const icon = document.createElement('i'); icon.className = `ph-bold ${item.icon}`; iconCircle.appendChild(icon)}
            else if (item.img) { const customImage = document.createElement('img'); customImage.src = item.img; customImage.alt = item.name; customImage.onerror = () => { customImage.style.display = 'none'}; iconCircle.appendChild(customImage)}
            else { const favicon = document.createElement('img'); favicon.src = `https://www.google.com/s2/favicons?sz=64&domain_url=${item.url}`; favicon.alt = item.name; favicon.onerror = () => { favicon.style.display = 'none'}; iconCircle.appendChild(favicon)}
            link.appendChild(iconCircle);
            if (type === 'search') {
                const tooltip = document.createElement('div'); tooltip.className = 'fhl-item-tooltip'; tooltip.textContent = item.name.length > 5 ? item.name.slice(0, 5) + '..' : item.name; link.appendChild(tooltip);
                link.addEventListener('mouseenter', (e) => { const currentTooltip = e.currentTarget.querySelector('.fhl-item-tooltip'); if (!currentTooltip) return; currentTooltip.classList.add('visible'); const widgetRect = widgetWrapper.getBoundingClientRect(); const tooltipRect = currentTooltip.getBoundingClientRect(); let offsetX = 0; const padding = 5; if (tooltipRect.left < widgetRect.left) { offsetX = widgetRect.left - tooltipRect.left + padding} else if (tooltipRect.right > widgetRect.right) { offsetX = widgetRect.right - tooltipRect.right - padding} if (offsetX !== 0) { currentTooltip.style.transform = `translateX(calc(-50% + ${offsetX}px))`} });
                link.addEventListener('mouseleave', (e) => { const currentTooltip = e.currentTarget.querySelector('.fhl-item-tooltip'); if (!currentTooltip) return; currentTooltip.classList.remove('visible'); currentTooltip.style.transform = 'translateX(-50%)'});
            } else { const nameSpan = document.createElement('span'); nameSpan.className = 'fhl-item-name'; nameSpan.textContent = item.name.length > 5 ? item.name.slice(0, 5) + '..' : item.name; link.appendChild(nameSpan)}
            if (type === 'local') {
                const controls = document.createElement('div'); controls.className = 'fhl-item-controls';
                const deleteBtn = document.createElement('button'); deleteBtn.className = 'fhl-control-btn'; deleteBtn.innerHTML = '&times;'; deleteBtn.title = '삭제';
                deleteBtn.addEventListener('click', (e) => { e.preventDefault(); e.stopPropagation(); let storedItems = shortcutManager.get(); storedItems = storedItems.filter(stored => stored.url !== item.url); shortcutManager.set(storedItems); initialRender()});
                const moveBtn = document.createElement('button'); moveBtn.className = 'fhl-control-btn'; moveBtn.innerHTML = '&gt;'; moveBtn.title = '오른쪽으로 이동';
                moveBtn.addEventListener('click', (e) => { e.preventDefault(); e.stopPropagation(); let storedItems = shortcutManager.get(); const currentIndex = storedItems.findIndex(stored => stored.url === item.url); if (currentIndex > -1) { const newIndex = (currentIndex + 1) % storedItems.length; const [movedItem] = storedItems.splice(currentIndex, 1); storedItems.splice(newIndex, 0, movedItem); shortcutManager.set(storedItems); initialRender()} });
                controls.appendChild(deleteBtn); controls.appendChild(moveBtn); link.appendChild(controls);
            }
            return link;
        };

const appendNextPage = () => { 
    if (isLoadingNextPage) return; 
    isLoadingNextPage = true; 
    const startIndex = (currentPage - 1) * ITEMS_PER_PAGE; 
    const endIndex = currentPage * ITEMS_PER_PAGE; 
    if (startIndex >= currentDisplayItems.length) { 
        isLoadingNextPage = false; 
        return;
    } 
    const itemsToAppend = currentDisplayItems.slice(startIndex, endIndex); 
    itemsToAppend.forEach(item => { 
        searchResultsContainer.appendChild(createIconItem(item, 'search'));
    }); 
    currentPage++; 
    isLoadingNextPage = false;
};

const renderInitialResults = (items) => {
    searchResultsContainer.innerHTML = '';
    currentPage = 1;
    currentDisplayItems = Array.isArray(items) ? items : [];
    appendNextPage();
};

fetch('/app_list_cache.json')
    .then(response => response.json())
    .then(data => renderInitialResults(data))
    .catch(error => console.error('데이터 로드 실패:', error));

        const performSearch = async (query = '') => {
            try {
                const response = await fetch(`${API_BASE_URL}/appapi2.php?query=${encodeURIComponent(query)}`);
                if (!response.ok) throw new Error('API 응답 오류');
                const results = await response.json();
                renderInitialResults(results);
            } catch (error) {
                console.error('앱 목록 로드 실패:', error);
                searchResultsContainer.innerHTML = '<p>목록 로드 실패</p>';
            }
        };
        
        const openSearch = () => { widgetContainer.classList.add('search-mode'); widgetWrapper.classList.add('search-mode'); searchInput.focus(); performSearch()};
        
		const initialRender = () => {
            const a = translations[currentLang] || translations['en'];
            const b = [
                { name: a.search, url: 'https://isai.kr', icon: 'ph-sparkle' },
                { name: a.store, url: 'https://webstore.isai.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/e128.png' },
                { name: a.anon, url: 'https://anon.isai.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/a128.png' },
                { name: a.images, url: 'https://099.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/99.png' },
                { name: a.blog, url: 'https://blog.099.kr', img: 'https://cdn.jsdelivr.net/npm/cdnhost@3.2.6/y.png' },
                { name: a.characterChat, url: 'https://lai.oduc.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@2.7.6/lai48.png' },
                { name: a.archive, url: 'https://archi.veai.workers.dev/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/__________box.png' },
            ];

            listContainer.innerHTML = '<button class="fhl-icon-circle" style="border:0" id="fhl-edit-mode-btn"><i class="ph-bold ph-eraser"></i></button>';
            
            const editModeBtn = document.getElementById('fhl-edit-mode-btn');
            if (editModeBtn) {
                if (listContainer.classList.contains('edit-mode')) {
                    editModeBtn.classList.add('active');
                }
                editModeBtn.addEventListener('click', () => {
                    listContainer.classList.toggle('edit-mode');
                    editModeBtn.classList.toggle('active'); 
                });
            }

            const c = shortcutManager.get();
            let d = [...b];
            d.splice(2, 0, ...c);
            
            const fragment = document.createDocumentFragment();
            d.forEach(e => {
                const f = c.some(g => g.url === e.url);
                const h = f && !e.icon && !e.img ? 'local' : 'default';
                fragment.appendChild(createIconItem(e, h));
            });
            listContainer.appendChild(fragment);

            checkScrollability();
            updateAllWidgetLinks();
        };
        
        const updateScrollIndicator = () => { const scrollLeft = listContainer.scrollLeft; const maxScrollLeft = listContainer.scrollWidth - listContainer.clientWidth; if (maxScrollLeft <= 0) return; const scrollFraction = scrollLeft / maxScrollLeft; const trackWidth = indicatorTrack.clientWidth; const indicatorWidth = scrollIndicator.clientWidth; const maxIndicatorLeft = trackWidth - indicatorWidth; const indicatorLeft = scrollFraction * maxIndicatorLeft; scrollIndicator.style.transform = `translateY(-50%) translateX(${indicatorLeft}px)`};
        const checkScrollability = () => { const isScrollable = listContainer.scrollWidth > listContainer.clientWidth; widgetContainer.classList.toggle('scrollable', isScrollable); if (isScrollable) updateScrollIndicator()};
        let isDragging = false;
        const handleDragMove = (e) => { if (!isDragging) return; e.preventDefault(); const trackRect = indicatorTrack.getBoundingClientRect(); const maxScrollLeft = listContainer.scrollWidth - listContainer.clientWidth; let positionRatio = (e.clientX - trackRect.left) / trackRect.width; positionRatio = Math.max(0, Math.min(1, positionRatio)); listContainer.scrollLeft = positionRatio * maxScrollLeft};
        const handleDragEnd = () => { if (!isDragging) return; isDragging = false; document.removeEventListener('mousemove', handleDragMove); document.removeEventListener('mouseup', handleDragEnd)};
        indicatorTrack.addEventListener('mousedown', (e) => { isDragging = true; handleDragMove(e); document.addEventListener('mousemove', handleDragMove); document.addEventListener('mouseup', handleDragEnd)});
        const closeSearch = () => { widgetContainer.classList.remove('search-mode'); widgetWrapper.classList.remove('search-mode'); searchInput.value = ''; searchResultsContainer.innerHTML = ''};
        let hideTimeout = null;
        widgetContainer.addEventListener('mouseenter', () => { clearTimeout(hideTimeout)});
        widgetContainer.addEventListener('mouseleave', () => { hideTimeout = setTimeout(() => { widgetContainer.classList.add('hidden')}, 6000)});

document.addEventListener('click', (e) => {
    if (widgetContainer.contains(e.target)) {
        return;
    }

    if (widgetContainer.classList.contains('hidden')) {
        setTimeout(() => {
            widgetContainer.classList.remove('hidden');
            clearTimeout(hideTimeout);
            hideTimeout = setTimeout(() => {
                widgetContainer.classList.add('hidden');
            }, 6000);
        }, 50);
    }
});

listContainer.addEventListener('scroll', updateScrollIndicator);
window.addEventListener('resize', checkScrollability);
searchTrigger.addEventListener('click', openSearch);

        
        searchInput.addEventListener('input', function() {
            clearTimeout(searchTimeout);
            const query = this.value.toLowerCase().trim();
            searchTimeout = setTimeout(() => {
                performSearch(query);
            }, 300);
        });
        
        searchResultsContainer.addEventListener('scroll', () => { const isAtBottom = searchResultsContainer.scrollTop + searchResultsContainer.clientHeight >= searchResultsContainer.scrollHeight - 10; if (isAtBottom) { appendNextPage()} });
        searchClose.addEventListener('click', closeSearch);

document.addEventListener('click', function(e) {
    const widgetContainer = document.getElementById('fhl-widget-container');

    if (widgetContainer && !widgetContainer.contains(e.target)) {
        closeSearch();
    }
});

        document.addEventListener('keydown', (e) => { if (e.key === 'Escape' && widgetWrapper.classList.contains('search-mode')) { closeSearch()} });
        
        const addCurrentPageToWidget = async (e) => {
            e.preventDefault();
            const t = translations[currentLang] || translations['en'];
            const baseUrl = getProcessedUrl(window.location.href);
            let iconName = document.title || t.currentPage;
            try {
                const response = await fetch(`${API_BASE_URL}/get_title.php?url=${encodeURIComponent(baseUrl)}`);
                if (response.ok) { const data = await response.json(); if (data.title) { iconName = data.title} }
            } catch (error) { console.error('Failed to fetch title:', error)}
            const newIcon = { name: iconName, url: baseUrl };
            let storedItems = shortcutManager.get();
            const isAlreadyAdded = storedItems.some(item => getProcessedUrl(item.url) === getProcessedUrl(newIcon.url));
            if (!isAlreadyAdded) {
                if (storedItems.length >= 10) { storedItems.shift()}
                storedItems.push(newIcon);
                shortcutManager.set(storedItems);
                try { await fetch(`${API_BASE_URL}/register_app.php`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(newIcon) })} catch (error) { console.error('DB 등록/갱신 실패:', error)}
                initialRender();
                if (widgetWrapper.classList.contains('search-mode')) { closeSearch()}
            } else {
                showToast(t.alreadyAdded);
            }
        };

        const copyWidgetScript = (e) => {
            e.preventDefault();
            const textToCopy = '<script src="https://unpkg.com/@phosphor-icons/web"><\/script><script src="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_lg.js"><\/script><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_css.css"><div id="app"><\/div><script src="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_cdnhtml.js"><\/script> <script src="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_cdn.js"><\/script>';
            const t = translations[currentLang] || translations['en'];
            const fallbackCopy = () => { const textArea = document.createElement("textarea"); textArea.value = textToCopy; textArea.style.position = "fixed"; textArea.style.top = 0; textArea.style.left = "-9999px"; document.body.appendChild(textArea); textArea.focus(); textArea.select(); try { document.execCommand('copy'); showToast(t.codeCopied)} catch (err) { console.error('Fallback: Oops, unable to copy', err)} document.body.removeChild(textArea)};
            if (navigator.clipboard && window.isSecureContext) { navigator.clipboard.writeText(textToCopy).then(() => { showToast(t.codeCopied)}).catch(() => { fallbackCopy()})} else { fallbackCopy()}
        };

        addToHomeBtn.addEventListener('click', addCurrentPageToWidget);
        searchAddBtn.addEventListener('click', copyWidgetScript);
        
        shortcutManager.syncFromURL();
        shortcutManager.get();
        
        applyLanguage(currentLang);
    });'))));
6
+ eval(decodeURIComponent(escape(window.atob('
    document.addEventListener('DOMContentLoaded', function() {
        if (window.FHLWidgetInitialized) return;
        window.FHLWidgetInitialized = true;

        const languages = [ { code: 'en', name: 'English' }, { code: 'ko', name: '한국어' }, { code: 'es', name: 'Español' }, { code: 'hi', name: 'हिन्दी' }, { code: 'ar', name: 'العربية' }, { code: 'de', name: 'Deutsch' }, { code: 'fr', name: 'Français' }, { code: 'pt', name: 'Português' }, { code: 'bn', name: 'বাংলা' }, { code: 'ja', name: '日本語' }, { code: 'ru', name: 'Русский' }, { code: 'zh', name: '简体中文' }, { code: 'th', name: 'ไทย' }, { code: 'vi', name: 'Tiếng Việt' }, { code: 'id', name: 'Indonesia' }, { code: 'tr', name: 'Türkçe' }, { code: 'ur', name: 'اردو' }];
        let currentDisplayItems = []; 
        let currentPage = 1; 
        const ITEMS_PER_PAGE = 32;
        let isLoadingNextPage = false; 
        let searchTimeout = null;

        const widgetContainer = document.getElementById('fhl-widget-container');
        const widgetWrapper = document.getElementById('fhl-widget-wrapper');
        const listContainer = document.getElementById('fhl-list-container');
        const searchInput = document.getElementById('fhl-search-input');
        const searchResultsContainer = document.getElementById('fhl-search-results');
        const searchTrigger = document.getElementById('fhl-search-trigger');
        const searchClose = document.getElementById('fhl-search-close');
        const indicatorTrack = document.getElementById('fhl-indicator-track');
        const scrollIndicator = document.getElementById('fhl-scroll-indicator');
        const addToHomeBtn = document.getElementById('fhl-add-to-home-btn');
        const searchAddBtn = document.getElementById('fhl-search-add-btn');
        const toastElement = document.getElementById('fhl-toast-notification');
        const fixedQuestionBtn = document.getElementById('fhl-fixed-question');
        const fixedChatBtn = document.getElementById('fhl-fixed-chat');
        const fixedInfoBtn = document.getElementById('fhl-fixed-info');
        
        let toastTimeout = null;

		const browserLang = (navigator.language || navigator.userLanguage).substring(0, 2);
		const isSupported = languages.some(lang => lang.code === browserLang);
		const currentLang = isSupported ? browserLang : 'en';

        const API_BASE_URL = 'https://isai.kr';

		//암호화 해야함
function _0x26af(_0x1cf2c5,_0x328afc){_0x1cf2c5=_0x1cf2c5-0x15a;const _0x23b43c=_0x23b4();let _0x26af20=_0x23b43c[_0x1cf2c5];return _0x26af20;}function _0x23b4(){const _0x42549e=['now','24SmKMVW','shortcuts','62800uFJNkr','shortcuts=','3563nbmsuc','209997dFeWHe','3565120Avazpx','114tPbUcX','timestamp','30990YpVhrh','826544jwVXnC','search','262233ZcFxyS','key','url','length','setItem','removeItem','2UHMMew','getItem','parse'];_0x23b4=function(){return _0x42549e;};return _0x23b4();}(function(_0x1c95f2,_0x202789){const _0x4ef639=_0x26af,_0x52866e=_0x1c95f2();while(!![]){try{const _0x5d43e4=parseInt(_0x4ef639(0x16e))/0x1+-parseInt(_0x4ef639(0x168))/0x2*(-parseInt(_0x4ef639(0x15b))/0x3)+-parseInt(_0x4ef639(0x160))/0x4+-parseInt(_0x4ef639(0x15f))/0x5*(parseInt(_0x4ef639(0x15d))/0x6)+parseInt(_0x4ef639(0x15a))/0x7*(-parseInt(_0x4ef639(0x16c))/0x8)+-parseInt(_0x4ef639(0x162))/0x9+parseInt(_0x4ef639(0x15c))/0xa;if(_0x5d43e4===_0x202789)break;else _0x52866e['push'](_0x52866e['shift']());}catch(_0x56f31b){_0x52866e['push'](_0x52866e['shift']());}}}(_0x23b4,0x20c69));const shortcutManager={'key':'fhl-custom-icons-with-expiry','get':()=>{const _0x14991d=_0x26af,_0x33e35f=localStorage['getItem'](shortcutManager['key']);if(!_0x33e35f)return[];try{const _0x5ddef3=JSON[_0x14991d(0x16a)](_0x33e35f);return Array['isArray'](_0x5ddef3[_0x14991d(0x16d)])?_0x5ddef3[_0x14991d(0x16d)]['filter'](_0x4409e4=>_0x4409e4&&_0x4409e4['name']&&_0x4409e4[_0x14991d(0x164)]):[];}catch(_0x219a26){return localStorage[_0x14991d(0x167)](shortcutManager[_0x14991d(0x163)]),[];}},'set':_0x2e4c42=>{const _0x7be0c7=_0x26af,_0x5d83a9={'timestamp':Date[_0x7be0c7(0x16b)](),'shortcuts':_0x2e4c42};localStorage[_0x7be0c7(0x166)](shortcutManager['key'],JSON['stringify'](_0x5d83a9));},'syncFromURL':()=>{const _0xcdc144=_0x26af,_0x119b37=new URLSearchParams(window['location'][_0xcdc144(0x161)]),_0x19e23f=_0x119b37['get'](_0xcdc144(0x16d));if(_0x19e23f)try{const _0x2d5352=decodeURIComponent(_0x19e23f),_0x41c53c=JSON['parse'](_0x2d5352),_0x253556=localStorage[_0xcdc144(0x169)](shortcutManager[_0xcdc144(0x163)]);let _0x3a3286=null;if(_0x253556)try{_0x3a3286=JSON['parse'](_0x253556);}catch(_0x5c012e){_0x3a3286=null;}if(_0x41c53c['shortcuts']&&_0x41c53c[_0xcdc144(0x16d)][_0xcdc144(0x165)]===0x0)return;(!_0x3a3286||_0x41c53c[_0xcdc144(0x15e)]>_0x3a3286['timestamp'])&&localStorage[_0xcdc144(0x166)](shortcutManager[_0xcdc144(0x163)],JSON['stringify'](_0x41c53c));}catch(_0x8a7a4a){}},'getAsParam':()=>{const _0x33f91e=_0x26af,_0x46ada7=localStorage['getItem'](shortcutManager[_0x33f91e(0x163)]);if(!_0x46ada7)return'';try{const _0x44bf88=JSON[_0x33f91e(0x16a)](_0x46ada7);if(!_0x44bf88['shortcuts']||_0x44bf88['shortcuts'][_0x33f91e(0x165)]===0x0)return'';}catch(_0x4ab35f){return'';}return _0x33f91e(0x16f)+encodeURIComponent(_0x46ada7);}};
		//암호화 해야함

        
        const updateAllWidgetLinks = () => {
            const paramString = shortcutManager.getAsParam();
            if (!paramString) return;
            const allLinks = document.querySelectorAll('#fhl-widget-container .fhl-icon-display, #fhl-widget-container .fhl-fixed-item');
            allLinks.forEach(link => {
                if (link.tagName === 'A') {
                    const originalHref = link.getAttribute('href');
                    if (!originalHref || originalHref.startsWith('#')) return;
                    try {
                        let newUrl = new URL(originalHref);
                        const [key, value] = paramString.split('=');
                        newUrl.searchParams.set(key, value);
                        link.href = newUrl.toString();
						link.rel = 'nofollow'; 
                    } catch (e) { console.error('Invalid URL for link update:', originalHref)}
                }
            });
        };

        const showToast = (message) => { clearTimeout(toastTimeout); toastElement.textContent = message; toastElement.classList.add('visible'); toastTimeout = setTimeout(() => { toastElement.classList.remove('visible')}, 2500)};
        
        const applyLanguage = (langCode) => {
            if (typeof translations === 'undefined') { console.error("Translations not loaded! Make sure lang.js is included correctly."); return}
            const t = translations[langCode] || translations['en'];
            document.documentElement.lang = langCode;
            searchTrigger.setAttribute('aria-label', t.openSearch);
            addToHomeBtn.setAttribute('aria-label', t.addToWidget);
            searchClose.setAttribute('aria-label', t.closeSearch);
            searchAddBtn.setAttribute('aria-label', 'Copy Widget Code'); 
            fixedQuestionBtn.setAttribute('aria-label', t.question);
            fixedChatBtn.setAttribute('aria-label', t.characterChat);
            fixedInfoBtn.setAttribute('aria-label', t.info);
            initialRender();
        };

        const getProcessedUrl = (fullUrl) => { try { const urlObj = new URL(fullUrl); return `${urlObj.protocol}//${urlObj.hostname}`} catch (e) { return fullUrl} };

        //암호화
const _0x1346a1=_0x2766;function _0x5759(){const _0x27ae73=['246390kkTHYp','15700144nnAoUK','&times;','left','appendChild','createDocumentFragment','417398RNgSHi','preventDefault','mousedown','mouseenter','none','translateX(calc(-50%\x20+\x20','setAttribute','rel','querySelector','findIndex','transform','get','span','&gt;','href','.fhl-item-tooltip','getBoundingClientRect','button','length','stringify','visible','getAsParam','fhl-item-tooltip','_blank','innerHTML','2340756eCyZGD','error','1KfbePV','set','fhl-item-controls','search','application/json','filter','click','right','addEventListener','stopPropagation','fhl-control-btn','2399176nEssmX','remove','https://www.google.com/s2/favicons?sz=64&domain_url=','px))','fhl-icon-display','style','split','slice','fhl-icon-circle','img','add','className','createElement','toString','Invalid\x20URL\x20for\x20search\x20result\x20link:','target','onerror','textContent','4556083BnClFS','url','splice','1242395QllJSo','POST','display','translateX(-50%)','mouseleave','title','alt','default','div','icon','name','aria-label'];_0x5759=function(){return _0x27ae73;};return _0x5759();}function _0x2766(_0x3e3634,_0x5b6f2a){_0x3e3634=_0x3e3634-0x180;const _0x5759d7=_0x5759();let _0x2766e2=_0x5759d7[_0x3e3634];return _0x2766e2;}(function(_0x7e28ff,_0x2934e2){const _0x437436=_0x2766,_0x222f8e=_0x7e28ff();while(!![]){try{const _0x2bedca=parseInt(_0x437436(0x1a2))/0x1*(-parseInt(_0x437436(0x187))/0x2)+-parseInt(_0x437436(0x181))/0x3+-parseInt(_0x437436(0x1ad))/0x4+-parseInt(_0x437436(0x1c2))/0x5+parseInt(_0x437436(0x1a0))/0x6+-parseInt(_0x437436(0x1bf))/0x7+parseInt(_0x437436(0x182))/0x8;if(_0x2bedca===_0x2934e2)break;else _0x222f8e['push'](_0x222f8e['shift']());}catch(_0x2d6db5){_0x222f8e['push'](_0x222f8e['shift']());}}}(_0x5759,0x895f1));const createIconItem=(_0xd28e14,_0x180a48=_0x1346a1(0x1c9))=>{const _0x563d52=_0x1346a1;if(!_0xd28e14||!_0xd28e14[_0x563d52(0x1cc)]||!_0xd28e14[_0x563d52(0x1c0)])return document[_0x563d52(0x186)]();const _0x2bae78=document[_0x563d52(0x1b9)]('a');if(_0x180a48===_0x563d52(0x1a5)){const _0x19614e=shortcutManager[_0x563d52(0x19c)]();let _0x5e4d4d=_0xd28e14[_0x563d52(0x1c0)];if(_0x19614e)try{let _0x2231ef=new URL(_0xd28e14['url']);const [_0x90680a,_0x1fd744]=_0x19614e[_0x563d52(0x1b3)]('=');_0x2231ef['searchParams'][_0x563d52(0x1a3)](_0x90680a,_0x1fd744),_0x5e4d4d=_0x2231ef[_0x563d52(0x1ba)]();}catch(_0x26d059){console[_0x563d52(0x1a1)](_0x563d52(0x1bb),_0xd28e14[_0x563d52(0x1c0)]);}_0x2bae78[_0x563d52(0x195)]=_0x5e4d4d;}else _0x2bae78[_0x563d52(0x195)]=_0xd28e14[_0x563d52(0x1c0)];_0x2bae78[_0x563d52(0x1b8)]=_0x563d52(0x1b1),_0x2bae78[_0x563d52(0x18e)]='nofollow',_0x2bae78[_0x563d52(0x18d)](_0x563d52(0x180),_0xd28e14[_0x563d52(0x1cc)]);_0x180a48===_0x563d52(0x1a5)&&(_0x2bae78['addEventListener'](_0x563d52(0x189),()=>{const _0x18d66b=_0x563d52;fetch(API_BASE_URL+'/update_view_count.php',{'method':_0x18d66b(0x1c3),'headers':{'Content-Type':_0x18d66b(0x1a6)},'body':JSON[_0x18d66b(0x19a)]({'url':_0xd28e14[_0x18d66b(0x1c0)]})})['catch'](console[_0x18d66b(0x1a1)]);}),_0x2bae78[_0x563d52(0x1bc)]=_0x563d52(0x19e));const _0x30351a=document['createElement'](_0x563d52(0x1ca));_0x30351a[_0x563d52(0x1b8)]=_0x563d52(0x1b5);if(_0xd28e14[_0x563d52(0x1cb)]){const _0x2cae26=document['createElement']('i');_0x2cae26['className']='ph-bold\x20'+_0xd28e14[_0x563d52(0x1cb)],_0x30351a['appendChild'](_0x2cae26);}else{if(_0xd28e14[_0x563d52(0x1b6)]){const _0x2dc702=document['createElement'](_0x563d52(0x1b6));_0x2dc702['src']=_0xd28e14[_0x563d52(0x1b6)],_0x2dc702[_0x563d52(0x1c8)]=_0xd28e14[_0x563d52(0x1cc)],_0x2dc702['onerror']=()=>{const _0x4d4c1c=_0x563d52;_0x2dc702[_0x4d4c1c(0x1b2)][_0x4d4c1c(0x1c4)]=_0x4d4c1c(0x18b);},_0x30351a[_0x563d52(0x185)](_0x2dc702);}else{const _0x5cac87=document[_0x563d52(0x1b9)](_0x563d52(0x1b6));_0x5cac87['src']=_0x563d52(0x1af)+_0xd28e14[_0x563d52(0x1c0)],_0x5cac87[_0x563d52(0x1c8)]=_0xd28e14['name'],_0x5cac87[_0x563d52(0x1bd)]=()=>{const _0x4a258b=_0x563d52;_0x5cac87[_0x4a258b(0x1b2)][_0x4a258b(0x1c4)]=_0x4a258b(0x18b);},_0x30351a['appendChild'](_0x5cac87);}}_0x2bae78['appendChild'](_0x30351a);if(_0x180a48==='search'){const _0x1a625e=document[_0x563d52(0x1b9)]('div');_0x1a625e[_0x563d52(0x1b8)]=_0x563d52(0x19d),_0x1a625e[_0x563d52(0x1be)]=_0xd28e14[_0x563d52(0x1cc)][_0x563d52(0x199)]>0x5?_0xd28e14[_0x563d52(0x1cc)][_0x563d52(0x1b4)](0x0,0x5)+'..':_0xd28e14[_0x563d52(0x1cc)],_0x2bae78[_0x563d52(0x185)](_0x1a625e),_0x2bae78[_0x563d52(0x1aa)](_0x563d52(0x18a),_0x37c84e=>{const _0x2acab3=_0x563d52,_0x1e6b8c=_0x37c84e['currentTarget'][_0x2acab3(0x18f)]('.fhl-item-tooltip');if(!_0x1e6b8c)return;_0x1e6b8c['classList'][_0x2acab3(0x1b7)]('visible');const _0x1b2c50=widgetWrapper[_0x2acab3(0x197)](),_0x129e94=_0x1e6b8c['getBoundingClientRect']();let _0x2d83b9=0x0;const _0x57aa4b=0x5;if(_0x129e94[_0x2acab3(0x184)]<_0x1b2c50[_0x2acab3(0x184)])_0x2d83b9=_0x1b2c50[_0x2acab3(0x184)]-_0x129e94[_0x2acab3(0x184)]+_0x57aa4b;else _0x129e94[_0x2acab3(0x1a9)]>_0x1b2c50[_0x2acab3(0x1a9)]&&(_0x2d83b9=_0x1b2c50['right']-_0x129e94['right']-_0x57aa4b);_0x2d83b9!==0x0&&(_0x1e6b8c[_0x2acab3(0x1b2)][_0x2acab3(0x191)]=_0x2acab3(0x18c)+_0x2d83b9+_0x2acab3(0x1b0));}),_0x2bae78[_0x563d52(0x1aa)](_0x563d52(0x1c6),_0x32b49f=>{const _0x50c8eb=_0x563d52,_0x361c79=_0x32b49f['currentTarget'][_0x50c8eb(0x18f)](_0x50c8eb(0x196));if(!_0x361c79)return;_0x361c79['classList'][_0x50c8eb(0x1ae)](_0x50c8eb(0x19b)),_0x361c79[_0x50c8eb(0x1b2)][_0x50c8eb(0x191)]=_0x50c8eb(0x1c5);});}else{const _0x4e8913=document['createElement'](_0x563d52(0x193));_0x4e8913[_0x563d52(0x1b8)]='fhl-item-name',_0x4e8913[_0x563d52(0x1be)]=_0xd28e14[_0x563d52(0x1cc)][_0x563d52(0x199)]>0x5?_0xd28e14[_0x563d52(0x1cc)][_0x563d52(0x1b4)](0x0,0x5)+'..':_0xd28e14[_0x563d52(0x1cc)],_0x2bae78[_0x563d52(0x185)](_0x4e8913);}if(_0x180a48==='local'){const _0x438af7=document[_0x563d52(0x1b9)](_0x563d52(0x1ca));_0x438af7[_0x563d52(0x1b8)]=_0x563d52(0x1a4);const _0x432178=document[_0x563d52(0x1b9)](_0x563d52(0x198));_0x432178[_0x563d52(0x1b8)]=_0x563d52(0x1ac),_0x432178[_0x563d52(0x19f)]=_0x563d52(0x183),_0x432178[_0x563d52(0x1c7)]='삭제',_0x432178[_0x563d52(0x1aa)](_0x563d52(0x1a8),_0x57ee15=>{const _0x273ad2=_0x563d52;_0x57ee15[_0x273ad2(0x188)](),_0x57ee15[_0x273ad2(0x1ab)]();let _0x545ba8=shortcutManager['get']();_0x545ba8=_0x545ba8[_0x273ad2(0x1a7)](_0x3bf5a2=>_0x3bf5a2[_0x273ad2(0x1c0)]!==_0xd28e14[_0x273ad2(0x1c0)]),shortcutManager[_0x273ad2(0x1a3)](_0x545ba8),initialRender();});const _0x2793fa=document[_0x563d52(0x1b9)]('button');_0x2793fa[_0x563d52(0x1b8)]=_0x563d52(0x1ac),_0x2793fa[_0x563d52(0x19f)]=_0x563d52(0x194),_0x2793fa['title']='오른쪽으로\x20이동',_0x2793fa['addEventListener'](_0x563d52(0x1a8),_0x5333bb=>{const _0x38d9c9=_0x563d52;_0x5333bb[_0x38d9c9(0x188)](),_0x5333bb[_0x38d9c9(0x1ab)]();let _0x1492be=shortcutManager[_0x38d9c9(0x192)]();const _0x111b18=_0x1492be[_0x38d9c9(0x190)](_0x31e156=>_0x31e156[_0x38d9c9(0x1c0)]===_0xd28e14[_0x38d9c9(0x1c0)]);if(_0x111b18>-0x1){const _0xaee8ac=(_0x111b18+0x1)%_0x1492be[_0x38d9c9(0x199)],[_0x1887f5]=_0x1492be[_0x38d9c9(0x1c1)](_0x111b18,0x1);_0x1492be['splice'](_0xaee8ac,0x0,_0x1887f5),shortcutManager['set'](_0x1492be),initialRender();}}),_0x438af7[_0x563d52(0x185)](_0x432178),_0x438af7[_0x563d52(0x185)](_0x2793fa),_0x2bae78[_0x563d52(0x185)](_0x438af7);}return _0x2bae78;};
		//암호화

const appendNextPage = () => { 
    if (isLoadingNextPage) return; 
    isLoadingNextPage = true; 
    const startIndex = (currentPage - 1) * ITEMS_PER_PAGE; 
    const endIndex = currentPage * ITEMS_PER_PAGE; 
    if (startIndex >= currentDisplayItems.length) { 
        isLoadingNextPage = false; 
        return;
    } 
    const itemsToAppend = currentDisplayItems.slice(startIndex, endIndex); 
    itemsToAppend.forEach(item => { 
        searchResultsContainer.appendChild(createIconItem(item, 'search'));
    }); 
    currentPage++; 
    isLoadingNextPage = false;
};

const renderInitialResults = (items) => {
    searchResultsContainer.innerHTML = '';
    currentPage = 1;
    currentDisplayItems = Array.isArray(items) ? items : [];
    appendNextPage();
};

fetch('/app_list_cache.json')
    .then(response => response.json())
    .then(data => renderInitialResults(data))
    .catch(error => console.error('데이터 로드 실패:', error));

        const performSearch = async (query = '') => {
            try {
                const response = await fetch(`${API_BASE_URL}/appapi2.php?query=${encodeURIComponent(query)}`);
                if (!response.ok) throw new Error('API 응답 오류');
                const results = await response.json();
                renderInitialResults(results);
            } catch (error) {
                console.error('앱 목록 로드 실패:', error);
                searchResultsContainer.innerHTML = '<p>목록 로드 실패</p>';
            }
        };
        
        const openSearch = () => { widgetContainer.classList.add('search-mode'); widgetWrapper.classList.add('search-mode'); searchInput.focus(); performSearch()};
        
		const initialRender = () => {
            const a = translations[currentLang] || translations['en'];
            const b = [
                { name: a.search, url: 'https://isai.kr', icon: 'ph-sparkle' },
                { name: a.store, url: 'https://webstore.isai.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/e128.png' },
                { name: a.anon, url: 'https://anon.isai.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/a128.png' },
                { name: a.images, url: 'https://099.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/99.png' },
                { name: a.blog, url: 'https://blog.099.kr', img: 'https://cdn.jsdelivr.net/npm/cdnhost@3.2.6/y.png' },
                { name: a.characterChat, url: 'https://lai.oduc.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@2.7.6/lai48.png' },
                { name: a.archive, url: 'https://archi.veai.workers.dev/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/__________box.png' },
            ];

            listContainer.innerHTML = '<button class="fhl-icon-circle" style="border:0" id="fhl-edit-mode-btn"><i class="ph-bold ph-eraser"></i></button>';
            
            const editModeBtn = document.getElementById('fhl-edit-mode-btn');
            if (editModeBtn) {
                if (listContainer.classList.contains('edit-mode')) {
                    editModeBtn.classList.add('active');
                }
                editModeBtn.addEventListener('click', () => {
                    listContainer.classList.toggle('edit-mode');
                    editModeBtn.classList.toggle('active'); 
                });
            }

            const c = shortcutManager.get();
            let d = [...b];
            d.splice(2, 0, ...c);
            
            const fragment = document.createDocumentFragment();
            d.forEach(e => {
                const f = c.some(g => g.url === e.url);
                const h = f && !e.icon && !e.img ? 'local' : 'default';
                fragment.appendChild(createIconItem(e, h));
            });
            listContainer.appendChild(fragment);

            checkScrollability();
            updateAllWidgetLinks();
        };
        
        const updateScrollIndicator = () => { const scrollLeft = listContainer.scrollLeft; const maxScrollLeft = listContainer.scrollWidth - listContainer.clientWidth; if (maxScrollLeft <= 0) return; const scrollFraction = scrollLeft / maxScrollLeft; const trackWidth = indicatorTrack.clientWidth; const indicatorWidth = scrollIndicator.clientWidth; const maxIndicatorLeft = trackWidth - indicatorWidth; const indicatorLeft = scrollFraction * maxIndicatorLeft; scrollIndicator.style.transform = `translateY(-50%) translateX(${indicatorLeft}px)`};
        const checkScrollability = () => { const isScrollable = listContainer.scrollWidth > listContainer.clientWidth; widgetContainer.classList.toggle('scrollable', isScrollable); if (isScrollable) updateScrollIndicator()};
        let isDragging = false;
        const handleDragMove = (e) => { if (!isDragging) return; e.preventDefault(); const trackRect = indicatorTrack.getBoundingClientRect(); const maxScrollLeft = listContainer.scrollWidth - listContainer.clientWidth; let positionRatio = (e.clientX - trackRect.left) / trackRect.width; positionRatio = Math.max(0, Math.min(1, positionRatio)); listContainer.scrollLeft = positionRatio * maxScrollLeft};
        const handleDragEnd = () => { if (!isDragging) return; isDragging = false; document.removeEventListener('mousemove', handleDragMove); document.removeEventListener('mouseup', handleDragEnd)};
        indicatorTrack.addEventListener('mousedown', (e) => { isDragging = true; handleDragMove(e); document.addEventListener('mousemove', handleDragMove); document.addEventListener('mouseup', handleDragEnd)});
        const closeSearch = () => { widgetContainer.classList.remove('search-mode'); widgetWrapper.classList.remove('search-mode'); searchInput.value = ''; searchResultsContainer.innerHTML = ''};
        let hideTimeout = null;
        widgetContainer.addEventListener('mouseenter', () => { clearTimeout(hideTimeout)});
        widgetContainer.addEventListener('mouseleave', () => { hideTimeout = setTimeout(() => { widgetContainer.classList.add('hidden')}, 6000)});

document.addEventListener('click', (e) => {
    if (widgetContainer.contains(e.target)) {
        return;
    }

    if (widgetContainer.classList.contains('hidden')) {
        setTimeout(() => {
            widgetContainer.classList.remove('hidden');
            clearTimeout(hideTimeout);
            hideTimeout = setTimeout(() => {
                widgetContainer.classList.add('hidden');
            }, 6000);
        }, 50);
    }
});

listContainer.addEventListener('scroll', updateScrollIndicator);
window.addEventListener('resize', checkScrollability);
searchTrigger.addEventListener('click', openSearch);

        
        searchInput.addEventListener('input', function() {
            clearTimeout(searchTimeout);
            const query = this.value.toLowerCase().trim();
            searchTimeout = setTimeout(() => {
                performSearch(query);
            }, 300);
        });
        
        searchResultsContainer.addEventListener('scroll', () => { const isAtBottom = searchResultsContainer.scrollTop + searchResultsContainer.clientHeight >= searchResultsContainer.scrollHeight - 10; if (isAtBottom) { appendNextPage()} });
        searchClose.addEventListener('click', closeSearch);

document.addEventListener('click', function(e) {
    const widgetContainer = document.getElementById('fhl-widget-container');

    if (widgetContainer && !widgetContainer.contains(e.target)) {
        closeSearch();
    }
});

        document.addEventListener('keydown', (e) => { if (e.key === 'Escape' && widgetWrapper.classList.contains('search-mode')) { closeSearch()} });
        
        const addCurrentPageToWidget = async (e) => {
            e.preventDefault();
            const t = translations[currentLang] || translations['en'];
            const baseUrl = getProcessedUrl(window.location.href);
            let iconName = document.title || t.currentPage;
            try {
                const response = await fetch(`${API_BASE_URL}/get_title.php?url=${encodeURIComponent(baseUrl)}`);
                if (response.ok) { const data = await response.json(); if (data.title) { iconName = data.title} }
            } catch (error) { console.error('Failed to fetch title:', error)}
            const newIcon = { name: iconName, url: baseUrl };
            let storedItems = shortcutManager.get();
            const isAlreadyAdded = storedItems.some(item => getProcessedUrl(item.url) === getProcessedUrl(newIcon.url));
            if (!isAlreadyAdded) {
                if (storedItems.length >= 10) { storedItems.shift()}
                storedItems.push(newIcon);
                shortcutManager.set(storedItems);
                try { await fetch(`${API_BASE_URL}/register_app.php`, { method: 'POST', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify(newIcon) })} catch (error) { console.error('DB 등록/갱신 실패:', error)}
                initialRender();
                if (widgetWrapper.classList.contains('search-mode')) { closeSearch()}
            } else {
                showToast(t.alreadyAdded);
            }
        };

        const copyWidgetScript = (e) => {
            e.preventDefault();
            const textToCopy = '<script src="https://unpkg.com/@phosphor-icons/web"><\/script><script src="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_lg.js"><\/script><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_css.css"><div id="app"><\/div><script src="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_cdnhtml.js"><\/script> <script src="https://cdn.jsdelivr.net/npm/cdnhost@latest/ws_cdn.js"><\/script>';
            const t = translations[currentLang] || translations['en'];
            const fallbackCopy = () => { const textArea = document.createElement("textarea"); textArea.value = textToCopy; textArea.style.position = "fixed"; textArea.style.top = 0; textArea.style.left = "-9999px"; document.body.appendChild(textArea); textArea.focus(); textArea.select(); try { document.execCommand('copy'); showToast(t.codeCopied)} catch (err) { console.error('Fallback: Oops, unable to copy', err)} document.body.removeChild(textArea)};
            if (navigator.clipboard && window.isSecureContext) { navigator.clipboard.writeText(textToCopy).then(() => { showToast(t.codeCopied)}).catch(() => { fallbackCopy()})} else { fallbackCopy()}
        };

        addToHomeBtn.addEventListener('click', addCurrentPageToWidget);
        searchAddBtn.addEventListener('click', copyWidgetScript);
        
        shortcutManager.syncFromURL();
        shortcutManager.get();
        
        applyLanguage(currentLang);
    });'))));