cdnhost 3.3.0 → 3.3.3
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.
- package/_newb.js +1 -1
- package/a1024.png +0 -0
- package/a128.png +0 -0
- package/aaa1.png +0 -0
- package/aaa2.png +0 -0
- package/link_google.js +1 -1
- package/package.json +1 -1
- package/ws_cdn.js +1 -1
- package/ws_css.css +216 -1
- package/isai.png +0 -0
- package/ss.png +0 -0
- package/ss2.png +0 -0
package/_newb.js
CHANGED
package/a1024.png
ADDED
|
Binary file
|
package/a128.png
ADDED
|
Binary file
|
package/aaa1.png
ADDED
|
Binary file
|
package/aaa2.png
ADDED
|
Binary file
|
package/link_google.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
eval(decodeURIComponent(escape(window.atob('
|
|
1
|
+
eval(decodeURIComponent(escape(window.atob('KGZ1bmN0aW9uICgpIHsNCmNvbnN0IHVybHMgPSBbImh0dHBzOi8vYTkuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTguMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTcuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTYuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTUuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTQuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vd3cuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTMuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTIuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYTEuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYWMuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYWIuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vYWEuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vNS4xeDgud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly82LjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovLzcuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vd3d3LmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2phLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2pwLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2phcGFuLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2EuZ2Zjb2luLndvcmtlcnMuZGV2LyIsImh0dHBzOi8veWFob28uZ2Zjb2luLndvcmtlcnMuZGV2LyIsImh0dHBzOi8veC5nZmNvaW4ud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly90d2l0dGVyLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL3dwLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL25ld3MuZ2Zjb2luLndvcmtlcnMuZGV2LyIsImh0dHBzOi8vOS4xeDgud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly84LjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL29vLjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL2Jsb2dnZXIuMXg4LndvcmtlcnMuZGV2LyIsImh0dHBzOi8vcGluLjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL2hpLjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL21zLjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL2ZhY2Vib29rLjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL2dvb2dsZS4xeDgud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly9uZXdzLjF4OC53b3JrZXJzLmRldi8iLCJodHRwczovL2hpLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovLzQuZ2Zjb2luLndvcmtlcnMuZGV2LyIsImh0dHBzOi8vcGluLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2tvLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL25ldC5nZmNvaW4ud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly8xMC5nZmNvaW4ud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly9ibG9nLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL3Bvc3QuZ2Zjb2luLndvcmtlcnMuZGV2LyIsImh0dHBzOi8vY29tLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2lvLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2phcGFuZXNlLmdmY29pbi53b3JrZXJzLmRldi8iLCJodHRwczovL2pwbi5nZmNvaW4ud29ya2Vycy5kZXYvIiwiaHR0cHM6Ly9nb29nbGUuZ2Zjb2luLndvcmtlcnMuZGV2LyJdOw0KDQogIGNvbnN0IGljb25zID0gWw0KICAgICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9jbG9jay1ob3VyLTUuc3ZnIiwgImh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vY2RuaG9zdEAyLjkuNi9pY29uL2Nsb2NrLWhlYXJ0LnN2ZyIsIA0KICAgICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9hcm1jaGFpci0yLnN2ZyIsICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9hcHAtd2luZG93LnN2ZyIsIA0KICAgICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9hbmdsZS5zdmciLCAiaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9jZG5ob3N0QDIuOS42L2ljb24vYWxpZ24tYm94LWxlZnQtbWlkZGxlLnN2ZyIsICANCiAgICAiaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9jZG5ob3N0QDIuOS42L2ljb24vY2xvdWQtYm9sdC5zdmciLCAiaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9jZG5ob3N0QDIuOS42L2ljb24vY2xvdWQtY29nLnN2ZyIsICANCiAgICAiaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9jZG5ob3N0QDIuOS42L2ljb24vY2xvdWQtYml0Y29pbi5zdmciLCAiaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9jZG5ob3N0QDIuOS42L2ljb24vY2xvdWQtaGVhcnQuc3ZnIiwgDQogICAgImh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vY2RuaG9zdEAyLjkuNi9pY29uL2Nsb3VkLWxvY2suc3ZnIiwgImh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vY2RuaG9zdEAyLjkuNi9pY29uL2NvZGUtYXN0ZXJpc2suc3ZnIiwgIA0KICAgICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9jb21wYXNzLnN2ZyIsICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9jb29raWUuc3ZnIiwgIA0KICAgICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9jb29raWUtbWFuLnN2ZyIsICJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL2Nkbmhvc3RAMi45LjYvaWNvbi9jdXAuc3ZnIg0KICBdOw0KDQogIGxldCBsaW5rc0h0bWwgPSAnJzsNCiAgdXJscy5mb3JFYWNoKCh1cmwsIGluZGV4KSA9PiB7DQogICAgY29uc3QgaWNvblVybCA9IGljb25zW2luZGV4ICUgaWNvbnMubGVuZ3RoXTsNCiAgICBjb25zdCBjbGVhbkRvbWFpbiA9IHVybC5yZXBsYWNlKC9eKGh0dHBzPzopP1wvXC8vLCAnJykucmVwbGFjZSgvXC8kLywgJycpOw0KICAgIGxpbmtzSHRtbCArPSBgPGEgaHJlZj0iJHt1cmx9IiB0YXJnZXQ9Il9ibGFuayIgdGl0bGU9IiR7Y2xlYW5Eb21haW59Ij48aW1nIHNyYz0iJHtpY29uVXJsfSIgYWx0PSJJY29uIj48L2E+YDsNCiAgfSk7DQoNCiAgY29uc3QgZnVsbEh0bWwgPSBgDQogICAgPGRpdiBpZD0iZ2xhc3MtYmFubmVyLXdyYXBwZXIiIHN0eWxlPSJ3aWR0aDogMTAwJTsgcGFkZGluZzogMjBweDsgYm94LXNpemluZzogYm9yZGVyLWJveDsgZGlzcGxheTogZmxleDsganVzdGlmeS1jb250ZW50OiBjZW50ZXI7Ij4NCiAgICAgIDxzdHlsZT4NCiAgICAgICAgLm5iYXItY29udGFpbmVyIHsNCiAgICAgICAgICBwb3NpdGlvbjogcmVsYXRpdmU7DQogICAgICAgICAgd2lkdGg6IDEwMCU7DQogICAgICAgICAgbWF4LXdpZHRoOiA4MDBweDsNCiAgICAgICAgICBoZWlnaHQ6IDQ4cHg7DQogICAgICAgICAgbWFyZ2luOiAwIGF1dG87DQogICAgICAgICAgYm9yZGVyLXJhZGl1czogMjRweDsNCiAgICAgICAgICBvdmVyZmxvdzogaGlkZGVuOw0KICAgICAgICAgIGJhY2tncm91bmQ6ICNmZmY7DQogICAgICAgICAgYm9yZGVyOiAxcHggc29saWQgcmdiYSgyNTUsIDI1NSwgMjU1LCAwLjMpOw0KICAgICAgICAgIGJveC1zaGFkb3c6IDAgOHB4IDMycHggMCByZ2JhKDMxLCAzOCwgMTM1LCAwLjEpOw0KICAgICAgICAgIGRpc3BsYXk6IGZsZXg7DQogICAgICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjsNCiAgICAgICAgfQ0KICAgICAgICAubmJhci1jb250YWluZXI6OmFmdGVyIHsNCiAgICAgICAgICBjb250ZW50OiAnJzsNCiAgICAgICAgICBwb3NpdGlvbjogYWJzb2x1dGU7DQogICAgICAgICAgdG9wOiAwOw0KICAgICAgICAgIHJpZ2h0OiA0NHB4Ow0KICAgICAgICAgIHdpZHRoOiA0MHB4Ow0KICAgICAgICAgIGhlaWdodDogMTAwJTsNCiAgICAgICAgICBiYWNrZ3JvdW5kOiBsaW5lYXItZ3JhZGllbnQodG8gcmlnaHQsIHJnYmEoMjU1LCAyNTUsIDI1NSwgMCksIHJnYmEoMjU1LCAyNTUsIDI1NSwgMSkgOTAlKTsNCiAgICAgICAgICBwb2ludGVyLWV2ZW50czogbm9uZTsNCiAgICAgICAgfQ0KICAgICAgICAubmJhci1zY3JvbGwtYXJlYSB7DQogICAgICAgICAgZmxleC1ncm93OiAxOw0KICAgICAgICAgIGRpc3BsYXk6IGZsZXg7DQogICAgICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjsNCiAgICAgICAgICBvdmVyZmxvdy14OiBhdXRvOw0KICAgICAgICAgIGhlaWdodDogMTAwJTsNCiAgICAgICAgICBwYWRkaW5nOiAwIDE1cHg7DQogICAgICAgICAgLW1zLW92ZXJmbG93LXN0eWxlOiBub25lOw0KICAgICAgICAgIHNjcm9sbGJhci13aWR0aDogbm9uZTsNCiAgICAgICAgfQ0KICAgICAgICAubmJhci1zY3JvbGwtYXJlYTo6LXdlYmtpdC1zY3JvbGxiYXIgew0KICAgICAgICAgIGRpc3BsYXk6IG5vbmU7DQogICAgICAgIH0NCiAgICAgICAgLm5iYXItc2Nyb2xsLWFyZWEgYSB7DQogICAgICAgICAgZGlzcGxheTogZmxleDsNCiAgICAgICAgICBhbGlnbi1pdGVtczogY2VudGVyOw0KICAgICAgICAgIGp1c3RpZnktY29udGVudDogY2VudGVyOw0KICAgICAgICAgIHRleHQtZGVjb3JhdGlvbjogbm9uZTsNCiAgICAgICAgICBwYWRkaW5nOiAwIDdweCAhaW1wb3J0YW50Ow0KICAgICAgICAgIGN1cnNvcjogcG9pbnRlcjsNCiAgICAgICAgICBvcGFjaXR5OiAwLjc7DQogICAgICAgIH0NCiAgICAgICAgLm5iYXItc2Nyb2xsLWFyZWEgYSBpbWcgew0KICAgICAgICAgIHdpZHRoOiAyMHB4ICFpbXBvcnRhbnQ7DQoJCSAgbWF4LXdpZHRoOiB1bnNldCAhaW1wb3J0YW50Ow0KICAgICAgICAgIGhlaWdodDogMjBweCAhaW1wb3J0YW50Ow0KICAgICAgICAgIGRpc3BsYXk6IGJsb2NrOw0KICAgICAgICB9DQogICAgICAgIC5uYmFyLXNjcm9sbC1hcmVhIGE6aG92ZXIgew0KICAgICAgICAgIG9wYWNpdHk6IDE7DQogICAgICAgIH0NCiAgICAgICAgLm5iYXItZml4ZWQtbWVudSB7DQogICAgICAgICAgZmxleC1zaHJpbms6IDA7DQogICAgICAgICAgd2lkdGg6IDQ0cHg7DQogICAgICAgICAgaGVpZ2h0OiAxMDAlOw0KICAgICAgICAgIGRpc3BsYXk6IGZsZXg7DQogICAgICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjsNCiAgICAgICAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjsNCiAgICAgICAgICBib3JkZXItbGVmdDogMXB4IHNvbGlkIHJnYmEoMjU1LDI1NSwyNTUsMC4yKTsNCiAgICAgICAgICBiYWNrZ3JvdW5kOiByZ2JhKDI1NSwyNTUsMjU1LDAuMDUpOw0KICAgICAgICAgIGN1cnNvcjogcG9pbnRlcjsNCiAgICAgICAgICB6LWluZGV4OiAyOw0KICAgICAgICB9DQogICAgICAgIC5uYmFyLWZpeGVkLW1lbnUgaW1nIHsNCiAgICAgICAgICB3aWR0aDogMThweDsNCiAgICAgICAgICBoZWlnaHQ6IDE4cHg7DQogICAgICAgICAgb3BhY2l0eTogMC44Ow0KICAgICAgICB9DQogICAgICA8L3N0eWxlPg0KICAgICAgPGRpdiBjbGFzcz0ibmJhci1jb250YWluZXIiPg0KICAgICAgICA8ZGl2IGNsYXNzPSJuYmFyLXNjcm9sbC1hcmVhIj4NCiAgICAgICAgICAke2xpbmtzSHRtbH0NCiAgICAgICAgPC9kaXY+DQogICAgICAgIDxkaXYgY2xhc3M9Im5iYXItZml4ZWQtbWVudSI+DQogICAgICAgICAgPGltZyBzcmM9Imh0dHBzOi8vZmlsZXMuc3ZnY2RuLmlvL2NvZGV4L21lbnUuc3ZnIiBhbHQ9Ik1lbnUiPg0KICAgICAgICA8L2Rpdj4NCiAgICAgIDwvZGl2Pg0KICAgIDwvZGl2Pg0KICBgOw0KICANCiAgZG9jdW1lbnQud3JpdGUoZnVsbEh0bWwpOw0KfSkoKTs='))));
|
package/package.json
CHANGED
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 languageBtn = document.getElementById('fhl-language-btn');
        const languagePanel = document.getElementById('fhl-language-panel');
        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.substring(0, 2);
		let currentLang = localStorage.getItem('fhl-widget-lang') || 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);
                    const thirtyDaysInMs = 30 * 24 * 60 * 60 * 1000;
                    if (Date.now() - data.timestamp > thirtyDaysInMs) {
                        localStorage.removeItem(shortcutManager.key);
                        return [];
                    }
                    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 decodedData = decodeURIComponent(shortcutsParam);
                        JSON.parse(decodedData);
                        localStorage.setItem(shortcutManager.key, decodedData);
                    } catch (e) { console.error("Failed to parse shortcuts from URL:", e)}
                }
            },
            getAsParam: () => {
                const rawData = localStorage.getItem(shortcutManager.key);
                if (!rawData) 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);
            languageBtn.setAttribute('aria-label', t.changeLanguage);
            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 populateLanguagePanel = () => { languagePanel.innerHTML = ''; languages.forEach(lang => { const langItem = document.createElement('a'); langItem.href = '#'; langItem.className = 'fhl-language-item'; langItem.textContent = lang.name; langItem.dataset.lang = lang.code; langItem.addEventListener('click', (e) => { e.preventDefault(); e.stopPropagation(); const newLang = e.target.dataset.lang; if (newLang !== currentLang) { currentLang = newLang; localStorage.setItem('fhl-widget-lang', currentLang); applyLanguage(currentLang)} languagePanel.classList.remove('visible')}); languagePanel.appendChild(langItem)})};

        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.question, url: 'https://isai.kr/#chat', icon: 'ph-question-mark' },
                { name: a.characterChat, url: 'https://lai.oduc.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@2.7.6/lai48.png' },
                { name: a.images, url: 'https://099.kr/', img: 'http://cdn.jsdelivr.net/npm/cdnhost@latest/99.png' },
                { name: a.blog, url: 'https://blog.099.kr', icon: 'ph-article-medium' },
                { name: 'Archive AI', url: 'https://archi.veai.workers.dev/', icon: 'ph-archive' },
            ];

            listContainer.innerHTML = '<button class="fhl-icon-circle" style="margin: 0 8px 0 0;border:0" id="fhl-edit-mode-btn"><i class="ph-bold ph-pencil-simple"></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 (!languagePanel.contains(e.target) && !languageBtn.contains(e.target)) { languagePanel.classList.remove('visible')} 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);
        languageBtn.addEventListener('click', (e) => { e.stopPropagation(); languagePanel.classList.toggle('visible')});

        
        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('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();
        
        populateLanguagePanel();
        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 languageBtn = document.getElementById('fhl-language-btn');
        const languagePanel = document.getElementById('fhl-language-panel');
        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.substring(0, 2);
		let currentLang = localStorage.getItem('fhl-widget-lang') || 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);
                    const thirtyDaysInMs = 30 * 24 * 60 * 60 * 1000;
                    if (Date.now() - data.timestamp > thirtyDaysInMs) {
                        localStorage.removeItem(shortcutManager.key);
                        return [];
                    }
                    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 decodedData = decodeURIComponent(shortcutsParam);
                        JSON.parse(decodedData);
                        localStorage.setItem(shortcutManager.key, decodedData);
                    } catch (e) { console.error("Failed to parse shortcuts from URL:", e)}
                }
            },
            getAsParam: () => {
                const rawData = localStorage.getItem(shortcutManager.key);
                if (!rawData) 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);
            languageBtn.setAttribute('aria-label', t.changeLanguage);
            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 populateLanguagePanel = () => { languagePanel.innerHTML = ''; languages.forEach(lang => { const langItem = document.createElement('a'); langItem.href = '#'; langItem.className = 'fhl-language-item'; langItem.textContent = lang.name; langItem.dataset.lang = lang.code; langItem.addEventListener('click', (e) => { e.preventDefault(); e.stopPropagation(); const newLang = e.target.dataset.lang; if (newLang !== currentLang) { currentLang = newLang; localStorage.setItem('fhl-widget-lang', currentLang); applyLanguage(currentLang)} languagePanel.classList.remove('visible')}); languagePanel.appendChild(langItem)})};

        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.question, url: 'https://isai.kr/#chat', icon: 'ph-question-mark' },
                { name: a.characterChat, url: 'https://lai.oduc.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@2.7.6/lai48.png' },
                { name: a.images, url: 'https://099.kr/', img: 'http://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: 'ANON', url: 'https://anon.isai.kr/', img: 'https://cdn.jsdelivr.net/npm/cdnhost@latest/a128.png' },
                { name: 'Archive AI', 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-pencil-simple"></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 (!languagePanel.contains(e.target) && !languageBtn.contains(e.target)) { languagePanel.classList.remove('visible')} 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);
        languageBtn.addEventListener('click', (e) => { e.stopPropagation(); languagePanel.classList.toggle('visible')});

        
        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('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();
        
        populateLanguagePanel();
        applyLanguage(currentLang);
    });'))));
|
package/ws_css.css
CHANGED
|
@@ -1 +1,216 @@
|
|
|
1
|
-
.fhl-widget-wrapper *, .fhl-widget-wrapper *::before, .fhl-widget-wrapper *::after { box-sizing: border-box; } #fhl-widget-container { position: fixed; z-index: 1000; bottom: 20px; left: 50%; transform: translateX(-50%) translateY(0); display: flex; flex-direction: column; align-items: center; width: calc(100% - 30px); max-width: 460px; pointer-events: auto; transition: opacity 0.4s ease-in-out, transform 0.4s ease-in-out; color: #000; font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif !important; line-height: 1.5 !important; box-sizing: border-box !important; } #fhl-widget-container.hidden { opacity: 0; transform: translateX(-50%) translateY(20px); pointer-events: none} #fhl-widget-container i { color: #000; font-size: 22px !important; font-style: normal; } #fhl-widget-container .fhl-item-tooltip { position: absolute; bottom: -28px; left: 50%; transform: translateX(-50%); background-color: #333; color: #fff; padding: 3px 6px; border-radius: 15px; font-size: 13px !important; font-weight: 600 !important; font-family: inherit !important; white-space: nowrap; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.2s ease-in-out, bottom 0.2s ease-in-out; z-index: 10; } #fhl-widget-container .fhl-indicator-track { position: relative; width: 100px; height: 14px; margin-bottom: 8px; cursor: pointer; background-color: transparent; transition: opacity 0.3s ease; opacity: 0; pointer-events: auto} #fhl-widget-container.scrollable .fhl-indicator-track { opacity: 1} #fhl-widget-container.search-mode .fhl-indicator-track { display: none} #fhl-widget-container .fhl-scroll-indicator { position: absolute; top: 50%; left: 0; width: 12px; height: 12px; background-color: #555; border-radius: 50%; transform: translateY(-50%); transition: transform 0.1s linear; pointer-events: none} #fhl-widget-container .fhl-widget-wrapper { position: relative; display: flex; align-items: center; width: 100%; max-width: 460px; height: 52px; padding: 7px 10px; border-radius: 26px; background-color: rgba(255, 255, 255, 0.5) !important; backdrop-filter: blur(25px) !important; -webkit-backdrop-filter: blur(25px) !important; box-shadow: 0 0 25px 5px rgba(255, 255, 255, 0.4), 0 8px 32px 0 rgba(31, 38, 135, 0.1); transition: all 0.3s ease-in-out; pointer-events: auto; overflow: visible; } #fhl-widget-container span { font-family: inherit !important; font-size: 13px !important; font-weight: 600 !important; font-style: normal !important; line-height: normal !important; letter-spacing: normal !important; text-align: left !important; text-decoration: none !important; text-transform: none !important; color: #333333 !important; background: transparent !important; background-color: transparent !important; border: none !important; border-radius: 0 !important; outline: none !important; margin: 0 !important; padding: 0 !important; box-shadow: none !important; text-shadow: none !important; box-sizing: border-box !important; display: inline-block !important; visibility: visible !important; opacity: 1 !important; float: none !important; } #fhl-widget-container .fhl-icon-display { position: relative; flex-shrink: 0; display: flex; justify-content: center; align-items: center; width: 38px; height: 38px; color: #333; background-color: rgba(0, 0, 0, 0.05); border: none; border-radius: 50%; text-decoration: none; cursor: pointer; padding: 0} #fhl-widget-container .fhl-icon-display:hover { background-color: rgba(0, 0, 0, 0.1)} #fhl-widget-container .fhl-list-container .fhl-icon-display { display: inline-flex; width: auto; padding: 0 8px 0 0; border-radius: 20px; background-color: transparent} #fhl-widget-container .fhl-list-container .fhl-icon-display:hover { background-color: rgba(0, 0, 0, 0.08)} #fhl-widget-container .fhl-list-container .fhl-icon-display:hover .fhl-icon-circle { background-color: transparent} #fhl-widget-container .fhl-item-name { max-width: 0; opacity: 0; transition: max-width 0.3s ease-in-out, opacity 0.2s ease-in-out 0.05s; white-space: nowrap; overflow: hidden; } #fhl-widget-container .fhl-list-container .fhl-icon-display:hover .fhl-item-name { max-width: 150px; opacity: 1} #fhl-widget-container .fhl-icon-circle { flex-shrink: 0; display: flex; justify-content: center; align-items: center; width: 38px; height: 38px; background-color: rgba(0, 0, 0, 0.05); border-radius: 50%; transition: background-color 0.2s ease; overflow: hidden} #fhl-widget-container .fhl-icon-circle img { width: 24px; height: 24px; object-fit: cover; border-radius: 50%} #fhl-widget-container .fhl-icon-bar { display: flex; align-items: center; gap: 8px; width: 100%; transition: opacity 0.3s ease} #fhl-widget-container .fhl-list-container { padding:0 8px; flex-grow: 1; display: flex; align-items: center; overflow-x: auto; -ms-overflow-style: none; scrollbar-width: none; scroll-snap-type: x mandatory; mask-image: linear-gradient(to right, transparent, black 20px, black calc(100% - 20px), transparent); -webkit-mask-image: linear-gradient(to right, transparent, black 20px, black calc(100% - 20px), transparent); padding-bottom: 30px; margin-bottom: -30px; overflow-y: visible} #fhl-widget-container .fhl-list-container::-webkit-scrollbar { display: none} #fhl-widget-container .fhl-list-container > .fhl-icon-display { scroll-snap-align: center} #fhl-widget-container .fhl-widget-wrapper.search-mode { height: 280px; align-items: flex-start; border-radius: 20px} #fhl-widget-container .fhl-widget-wrapper.search-mode .fhl-icon-bar { opacity: 0; pointer-events: none} #fhl-widget-container .fhl-search-view { position: absolute; top: 0; left: 0; width: 100%; height: 100%; display: flex; flex-direction: column; padding: 10px; opacity: 0; visibility: hidden; transition: opacity 0.3s 0.1s ease} #fhl-widget-container .fhl-widget-wrapper.search-mode .fhl-search-view { opacity: 1; visibility: visible} #fhl-widget-container .fhl-search-input-wrapper { display: flex; align-items: center; flex-shrink: 0; margin-bottom: 12px} #fhl-widget-container .fhl-search-input { flex-grow: 1; height: 38px; border: none; background: transparent; font-size: 16px; color: #111; padding: 0 10px; outline: none; transition: border-color 0.2s ease} #fhl-widget-container .fhl-search-input:focus { border-bottom-color: #007bff} #fhl-widget-container .fhl-search-content { display: grid; grid-template-columns: 1fr 38px; flex-grow: 1; overflow: hidden; } #fhl-widget-container .fhl-search-results { padding: 0 0 0 5px; display: flex; align-items: center; justify-content: center; flex-wrap: wrap; align-content: flex-start; gap: 10px; min-width: 0; overflow-y: auto; overflow-x: hidden; padding:0 10px; box-sizing:bordeer-box; } #fhl-widget-container .fhl-search-results::-webkit-scrollbar { display: none} #fhl-widget-container .fhl-search-results .fhl-icon-display { background-color: transparent} #fhl-widget-container .fhl-search-results .fhl-icon-display:hover .fhl-icon-circle { background-color: rgba(0, 0, 0, 0.1)} #fhl-widget-container .fhl-fixed-panel { display: flex; flex-direction: column; align-items: center; gap: 10px; } #fhl-widget-container .fhl-fixed-item { display: flex; align-items: center; justify-content: center; width: 38px !important; height: 38px !important; border-radius: 50% !important; background-color: rgba(0, 0, 0, 0.05) !important; text-decoration: none; cursor: pointer; transition: background-color 0.2s ease; flex-shrink: 0; } #fhl-widget-container .fhl-fixed-item:hover { background-color: rgba(0, 0, 0, 0.1) !important; } #fhl-widget-container .fhl-fixed-item i { color: #333 !important; font-size: 20px !important; } #fhl-widget-container .fhl-fixed-item img { width: 24px !important; height: 24px !important; border-radius: 50%; object-fit: cover; display: block; } #fhl-widget-container .fhl-item-tooltip { position: absolute; bottom: -28px; left: 50%; transform: translateX(-50%); background-color: #333; color: #fff; padding: 3px 6px; border-radius: 15px; font-size: 11px !important; font-weight: 500; white-space: nowrap; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.2s ease-in-out, bottom 0.2s ease-in-out; z-index: 10} #fhl-widget-container .fhl-item-tooltip.visible { opacity: 1; visibility: visible; bottom: -25px} #fhl-widget-container .fhl-item-controls { position: absolute; bottom: -20px; left: 20px; transform: translateX(-50%); display: flex; gap: 4px; background-color: rgba(0,0,0,0.7); border-radius: 10px; padding: 2px 4px; opacity: 0; visibility: hidden; transition: all 0.2s ease; pointer-events: none; z-index: 20} #fhl-widget-container .fhl-list-container.edit-mode .fhl-item-controls { opacity: 1; visibility: visible; pointer-events: auto; bottom: -13px} #fhl-widget-container .fhl-control-btn { background: none; border: none; color: white; cursor: pointer; font-size: 16px; line-height: 1; padding: 2px; font-family: monospace} #fhl-widget-container .fhl-control-btn:hover { color: #007bff} .fhl-toast { position: fixed; bottom: 90px; left: 50%; transform: translateX(-50%) translateY(10px); background-color: rgba(0, 0, 0, 0.8); color: #fff; padding: 8px 16px; border-radius: 18px; font-size: 14px; z-index: 1001; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.3s ease, transform 0.3s ease, visibility 0.3s} .fhl-toast.visible { opacity: 1; visibility: visible; transform: translateX(-50%) translateY(0)} #fhl-widget-container .fhl-language-panel { position: absolute; bottom: 100%; left: 50%; transform: translateX(-50%); margin-bottom: 15px; background-color: rgba(255, 255, 255, 0.4) !important; backdrop-filter: blur(25px) !important; -webkit-backdrop-filter: blur(25px) !important; border-radius: 25px; box-shadow: 0 0 25px 5px rgba(255, 255, 255, 0.4), 0 8px 32px 0 rgba(31, 38, 135, 0.1); padding: 12px; display: grid; grid-template-columns: 1fr 1fr; gap: 4px; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.3s ease, transform 0.3s ease; transform-origin: bottom center; } #fhl-widget-container .fhl-language-panel.visible { opacity: 1; visibility: visible; pointer-events: auto; } #fhl-widget-container .fhl-language-item { display: block; padding: 8px 16px; font-size: 14px; font-weight: 500; color: rgba(0, 0, 0, 0.6); text-decoration: none; border-radius: 20px; transition: all 0.2s ease; cursor: pointer; text-align: center; } #fhl-widget-container .fhl-language-item:hover { background-color: rgba(255, 255, 255, 0.3); color: rgba(0, 0, 0, 0.9); }
|
|
1
|
+
.fhl-widget-wrapper *, .fhl-widget-wrapper *::before, .fhl-widget-wrapper *::after {
|
|
2
|
+
box-sizing: border-box;
|
|
3
|
+
}
|
|
4
|
+
|
|
5
|
+
#fhl-widget-container {
|
|
6
|
+
position: fixed;
|
|
7
|
+
z-index: 1000;
|
|
8
|
+
bottom: 20px;
|
|
9
|
+
left: 50%;
|
|
10
|
+
transform: translateX(-50%) translateY(0);
|
|
11
|
+
display: flex;
|
|
12
|
+
flex-direction: column;
|
|
13
|
+
align-items: center;
|
|
14
|
+
width: calc(100% - 30px);
|
|
15
|
+
max-width: 460px;
|
|
16
|
+
pointer-events: auto;
|
|
17
|
+
transition: opacity 0.4s ease-in-out, transform 0.4s ease-in-out;
|
|
18
|
+
color: #000;
|
|
19
|
+
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif !important;
|
|
20
|
+
line-height: 1.5 !important;
|
|
21
|
+
box-sizing: border-box !important;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
#fhl-widget-container.hidden { opacity: 0; transform: translateX(-50%) translateY(20px); pointer-events: none}
|
|
25
|
+
|
|
26
|
+
#fhl-widget-container i {
|
|
27
|
+
color: #000;
|
|
28
|
+
font-size: 22px !important;
|
|
29
|
+
font-style: normal;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
#fhl-widget-container .fhl-item-tooltip {
|
|
33
|
+
position: absolute;
|
|
34
|
+
bottom: -28px;
|
|
35
|
+
left: 50%;
|
|
36
|
+
transform: translateX(-50%);
|
|
37
|
+
background-color: #333;
|
|
38
|
+
color: #fff;
|
|
39
|
+
padding: 3px 6px;
|
|
40
|
+
border-radius: 15px;
|
|
41
|
+
font-size: 13px !important;
|
|
42
|
+
font-weight: 600 !important;
|
|
43
|
+
font-family: inherit !important;
|
|
44
|
+
white-space: nowrap;
|
|
45
|
+
opacity: 0;
|
|
46
|
+
visibility: hidden;
|
|
47
|
+
pointer-events: none;
|
|
48
|
+
transition: opacity 0.2s ease-in-out, bottom 0.2s ease-in-out;
|
|
49
|
+
z-index: 10;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
#fhl-widget-container .fhl-indicator-track { position: relative; width: 100px; height: 14px; margin-bottom: 8px; cursor: pointer; background-color: transparent; transition: opacity 0.3s ease; opacity: 0; pointer-events: auto}
|
|
53
|
+
#fhl-widget-container.scrollable .fhl-indicator-track { opacity: 1}
|
|
54
|
+
#fhl-widget-container.search-mode .fhl-indicator-track { display: none}
|
|
55
|
+
#fhl-widget-container .fhl-scroll-indicator { position: absolute; top: 50%; left: 0; width: 12px; height: 12px; background-color: #555; border-radius: 50%; transform: translateY(-50%); transition: transform 0.1s linear; pointer-events: none}
|
|
56
|
+
|
|
57
|
+
#fhl-widget-container .fhl-widget-wrapper {
|
|
58
|
+
position: relative;
|
|
59
|
+
display: flex;
|
|
60
|
+
align-items: center;
|
|
61
|
+
width: 100%;
|
|
62
|
+
max-width: 460px;
|
|
63
|
+
height: 52px;
|
|
64
|
+
padding: 7px 10px;
|
|
65
|
+
border-radius: 26px;
|
|
66
|
+
|
|
67
|
+
background-color: rgba(255, 255, 255, 0.5) !important;
|
|
68
|
+
|
|
69
|
+
backdrop-filter: blur(25px) !important;
|
|
70
|
+
-webkit-backdrop-filter: blur(25px) !important;
|
|
71
|
+
|
|
72
|
+
box-shadow:
|
|
73
|
+
0 0 25px 5px rgba(255, 255, 255, 0.4),
|
|
74
|
+
0 8px 32px 0 rgba(31, 38, 135, 0.1);
|
|
75
|
+
|
|
76
|
+
transition: all 0.3s ease-in-out;
|
|
77
|
+
pointer-events: auto;
|
|
78
|
+
overflow: visible;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
#fhl-widget-container span {
|
|
82
|
+
font-family: inherit !important;
|
|
83
|
+
font-size: 13px !important;
|
|
84
|
+
font-weight: 600 !important;
|
|
85
|
+
font-style: normal !important;
|
|
86
|
+
line-height: normal !important;
|
|
87
|
+
letter-spacing: normal !important;
|
|
88
|
+
text-align: left !important;
|
|
89
|
+
text-decoration: none !important;
|
|
90
|
+
text-transform: none !important;
|
|
91
|
+
color: #333333 !important;
|
|
92
|
+
background: transparent !important;
|
|
93
|
+
background-color: transparent !important;
|
|
94
|
+
border: none !important;
|
|
95
|
+
border-radius: 0 !important;
|
|
96
|
+
outline: none !important;
|
|
97
|
+
margin: 0 !important;
|
|
98
|
+
padding: 0 !important;
|
|
99
|
+
box-shadow: none !important;
|
|
100
|
+
text-shadow: none !important;
|
|
101
|
+
box-sizing: border-box !important;
|
|
102
|
+
display: inline-block !important;
|
|
103
|
+
visibility: visible !important;
|
|
104
|
+
opacity: 1 !important;
|
|
105
|
+
float: none !important;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
#fhl-widget-container .fhl-icon-display { position: relative; flex-shrink: 0; display: flex; justify-content: center; align-items: center; width: 38px; height: 38px; color: #333; background-color: rgba(0, 0, 0, 0.05); border: none; border-radius: 50%; text-decoration: none; cursor: pointer; padding: 0}
|
|
109
|
+
#fhl-widget-container .fhl-icon-display:hover { background-color: rgba(0, 0, 0, 0.1);}
|
|
110
|
+
|
|
111
|
+
#fhl-widget-container .fhl-list-container .fhl-icon-display {display: inline-flex; width: auto; border-radius: 20px; background-color: transparent}
|
|
112
|
+
#fhl-widget-container .fhl-list-container .fhl-icon-display:hover {padding:0 8px 0 0;box-sizing:border-box;background-color: rgba(0, 0, 0, 0.08)}
|
|
113
|
+
#fhl-widget-container .fhl-list-container .fhl-icon-display:hover .fhl-icon-circle { background-color: transparent}
|
|
114
|
+
|
|
115
|
+
#fhl-widget-container .fhl-item-name { max-width: 0; opacity: 0; transition: max-width 0.3s ease-in-out, opacity 0.2s ease-in-out 0.05s; white-space: nowrap; overflow: hidden; }
|
|
116
|
+
#fhl-widget-container .fhl-list-container .fhl-icon-display:hover .fhl-item-name { max-width: 150px; opacity: 1}
|
|
117
|
+
|
|
118
|
+
#fhl-widget-container .fhl-icon-circle { flex-shrink: 0; display: flex; justify-content: center; align-items: center; width: 38px; height: 38px; background-color: rgba(0, 0, 0, 0.05); border-radius: 50%; transition: background-color 0.2s ease; overflow: hidden}
|
|
119
|
+
#fhl-widget-container .fhl-icon-circle img { width: 24px; height: 24px; object-fit: cover; border-radius: 50%}
|
|
120
|
+
|
|
121
|
+
#fhl-widget-container .fhl-icon-bar { display: flex; align-items: center; gap: 6px; width: 100%; transition: opacity 0.3s ease}
|
|
122
|
+
#fhl-widget-container .fhl-list-container { padding:0 8px; flex-grow: 1; display: flex; align-items: center; gap: 6px; overflow-x: auto; -ms-overflow-style: none; scrollbar-width: none; scroll-snap-type: x mandatory; mask-image: linear-gradient(to right, transparent, black 20px, black calc(100% - 20px), transparent); -webkit-mask-image: linear-gradient(to right, transparent, black 20px, black calc(100% - 20px), transparent); padding-bottom: 30px; margin-bottom: -30px; overflow-y: visible}
|
|
123
|
+
#fhl-widget-container .fhl-list-container::-webkit-scrollbar { display: none}
|
|
124
|
+
#fhl-widget-container .fhl-list-container > .fhl-icon-display { scroll-snap-align: center}
|
|
125
|
+
|
|
126
|
+
#fhl-widget-container .fhl-widget-wrapper.search-mode { height: 280px; align-items: flex-start; border-radius: 20px}
|
|
127
|
+
#fhl-widget-container .fhl-widget-wrapper.search-mode .fhl-icon-bar { opacity: 0; pointer-events: none}
|
|
128
|
+
|
|
129
|
+
#fhl-widget-container .fhl-search-view { position: absolute; top: 0; left: 0; width: 100%; height: 100%; display: flex; flex-direction: column; padding: 10px; opacity: 0; visibility: hidden; transition: opacity 0.3s 0.1s ease}
|
|
130
|
+
#fhl-widget-container .fhl-widget-wrapper.search-mode .fhl-search-view { opacity: 1; visibility: visible}
|
|
131
|
+
|
|
132
|
+
#fhl-widget-container .fhl-search-input-wrapper { display: flex; align-items: center; flex-shrink: 0; margin-bottom: 12px}
|
|
133
|
+
#fhl-widget-container .fhl-search-input { flex-grow: 1; height: 38px; border: none; background: transparent; font-size: 16px; color: #111; padding: 0 10px; outline: none; transition: border-color 0.2s ease}
|
|
134
|
+
#fhl-widget-container .fhl-search-input:focus { border-bottom-color: #007bff}
|
|
135
|
+
|
|
136
|
+
#fhl-widget-container .fhl-search-content {
|
|
137
|
+
display: grid;
|
|
138
|
+
grid-template-columns: 1fr 38px;
|
|
139
|
+
flex-grow: 1;
|
|
140
|
+
overflow: hidden;
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
#fhl-widget-container .fhl-search-results {
|
|
144
|
+
padding: 0 0 0 5px;
|
|
145
|
+
display: flex;
|
|
146
|
+
align-items: center;
|
|
147
|
+
justify-content: center;
|
|
148
|
+
flex-wrap: wrap;
|
|
149
|
+
align-content: flex-start;
|
|
150
|
+
gap: 10px;
|
|
151
|
+
min-width: 0;
|
|
152
|
+
overflow-y: auto;
|
|
153
|
+
overflow-x: hidden;
|
|
154
|
+
padding:0 10px;
|
|
155
|
+
box-sizing:bordeer-box;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
#fhl-widget-container .fhl-search-results::-webkit-scrollbar { display: none}
|
|
159
|
+
#fhl-widget-container .fhl-search-results .fhl-icon-display { background-color: transparent}
|
|
160
|
+
#fhl-widget-container .fhl-search-results .fhl-icon-display:hover .fhl-icon-circle { background-color: rgba(0, 0, 0, 0.1)}
|
|
161
|
+
|
|
162
|
+
#fhl-widget-container .fhl-fixed-panel {
|
|
163
|
+
display: flex;
|
|
164
|
+
flex-direction: column;
|
|
165
|
+
align-items: center;
|
|
166
|
+
gap: 10px;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
#fhl-widget-container .fhl-fixed-item {
|
|
170
|
+
display: flex;
|
|
171
|
+
align-items: center;
|
|
172
|
+
justify-content: center;
|
|
173
|
+
width: 38px !important;
|
|
174
|
+
height: 38px !important;
|
|
175
|
+
border-radius: 50% !important;
|
|
176
|
+
background-color: rgba(0, 0, 0, 0.05) !important;
|
|
177
|
+
text-decoration: none;
|
|
178
|
+
cursor: pointer;
|
|
179
|
+
transition: background-color 0.2s ease;
|
|
180
|
+
flex-shrink: 0;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
#fhl-widget-container .fhl-fixed-item:hover {
|
|
184
|
+
background-color: rgba(0, 0, 0, 0.1) !important;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
#fhl-widget-container .fhl-fixed-item i {
|
|
188
|
+
color: #333 !important;
|
|
189
|
+
font-size: 20px !important;
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
#fhl-widget-container .fhl-fixed-item img {
|
|
193
|
+
width: 24px !important;
|
|
194
|
+
height: 24px !important;
|
|
195
|
+
border-radius: 50%;
|
|
196
|
+
object-fit: cover;
|
|
197
|
+
display: block;
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
#fhl-widget-container .fhl-item-tooltip { position: absolute; bottom: -28px; left: 50%; transform: translateX(-50%); background-color: #333; color: #fff; padding: 3px 6px; border-radius: 15px; font-size: 11px !important; font-weight: 500; white-space: nowrap; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.2s ease-in-out, bottom 0.2s ease-in-out; z-index: 10}
|
|
201
|
+
#fhl-widget-container .fhl-item-tooltip.visible { opacity: 1; visibility: visible; bottom: -25px}
|
|
202
|
+
|
|
203
|
+
#fhl-widget-container .fhl-item-controls { position: absolute; bottom: -20px; left: 20px; transform: translateX(-50%); display: flex; gap: 4px; background-color: rgba(0,0,0,0.7); border-radius: 10px; padding: 2px 4px; opacity: 0; visibility: hidden; transition: all 0.2s ease; pointer-events: none; z-index: 20}
|
|
204
|
+
#fhl-widget-container .fhl-list-container.edit-mode .fhl-item-controls { opacity: 1; visibility: visible; pointer-events: auto; bottom: -13px}
|
|
205
|
+
|
|
206
|
+
#fhl-widget-container .fhl-control-btn { background: none; border: none; color: white; cursor: pointer; font-size: 16px; line-height: 1; padding: 2px; font-family: monospace}
|
|
207
|
+
#fhl-widget-container .fhl-control-btn:hover { color: #007bff}
|
|
208
|
+
|
|
209
|
+
.fhl-toast { position: fixed; bottom: 90px; left: 50%; transform: translateX(-50%) translateY(10px); background-color: rgba(0, 0, 0, 0.8); color: #fff; padding: 8px 16px; border-radius: 18px; font-size: 14px; z-index: 1001; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.3s ease, transform 0.3s ease, visibility 0.3s}
|
|
210
|
+
.fhl-toast.visible { opacity: 1; visibility: visible; transform: translateX(-50%) translateY(0)}
|
|
211
|
+
|
|
212
|
+
#fhl-widget-container .fhl-language-panel { position: absolute; bottom: 100%; left: 50%; transform: translateX(-50%); margin-bottom: 15px; background-color: rgba(255, 255, 255, 0.4) !important; backdrop-filter: blur(25px) !important; -webkit-backdrop-filter: blur(25px) !important; border-radius: 25px; box-shadow: 0 0 25px 5px rgba(255, 255, 255, 0.4), 0 8px 32px 0 rgba(31, 38, 135, 0.1); padding: 12px; display: grid; grid-template-columns: 1fr 1fr; gap: 4px; opacity: 0; visibility: hidden; pointer-events: none; transition: opacity 0.3s ease, transform 0.3s ease; transform-origin: bottom center; }
|
|
213
|
+
#fhl-widget-container .fhl-language-panel.visible { opacity: 1; visibility: visible; pointer-events: auto; }
|
|
214
|
+
|
|
215
|
+
#fhl-widget-container .fhl-language-item { display: block; padding: 8px 16px; font-size: 14px; font-weight: 500; color: rgba(0, 0, 0, 0.6); text-decoration: none; border-radius: 20px; transition: all 0.2s ease; cursor: pointer; text-align: center; }
|
|
216
|
+
#fhl-widget-container .fhl-language-item:hover { background-color: rgba(255, 255, 255, 0.3); color: rgba(0, 0, 0, 0.9); }
|
package/isai.png
DELETED
|
Binary file
|
package/ss.png
DELETED
|
Binary file
|
package/ss2.png
DELETED
|
Binary file
|