\n \n \n \n`;\n\nconst REWARDED_AD_THANKS_CSS = css`\n ${DEFAULT_BUTTON}\n ${REWARDED_AD_CLOSE_BUTTON_CSS}\n ${REWARDED_AD_PROMPT}\n ${EXIT_CSS}\n\n .rewarded-ad-prompt {\n width: 100%;\n }\n\n .rewarded-ad-thanks-message {\n font-size: 22px; // 1.375rem;\n font-weight: 400;\n line-height: 28px; // 1.75rem;\n letter-spacing: 0px;\n text-align: center;\n color: #202124;\n margin-block-end: 48px; // 3rem;\n }\n`;\n\nexport const REWARDED_AD_THANKS_HTML = html`\n \n
\n
${REWARDED_AD_CLOSE_BUTTON_HTML}
\n
\n $THANKS_FOR_VIEWING_THIS_AD$\n
\n \n`;\n","/**\n * Copyright 2024 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Intervention types that can be returned from the article endpoint.\n */\nexport enum InterventionType {\n TYPE_REGISTRATION_WALL = 'TYPE_REGISTRATION_WALL',\n TYPE_NEWSLETTER_SIGNUP = 'TYPE_NEWSLETTER_SIGNUP',\n TYPE_REWARDED_SURVEY = 'TYPE_REWARDED_SURVEY',\n TYPE_REWARDED_AD = 'TYPE_REWARDED_AD',\n TYPE_CONTRIBUTION = 'TYPE_CONTRIBUTION',\n TYPE_SUBSCRIPTION = 'TYPE_SUBSCRIPTION',\n TYPE_BYO_CTA = 'TYPE_BYO_CTA',\n}\n","/**\n * Copyright 2021 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n// Why is there a strings.ts and a swg-strings.ts, you ask? strings.ts is a\n// generated file, currently used for gaa builds. This file is used for swg and\n// swg-basic builds, and is currently manually updated.\n// TODO(stellachui): Figure out if they should be merged without a large impact\n// on binary size.\n\nexport const SWG_I18N_STRINGS = {\n 'SUBSCRIPTION_TITLE_LANG_MAP': {\n 'en': 'Subscribe with Google',\n 'ar': 'Google اشترك مع',\n 'de': 'Abonnieren mit Google',\n 'en-au': 'Subscribe with Google',\n 'en-ca': 'Subscribe with Google',\n 'en-gb': 'Subscribe with Google',\n 'en-us': 'Subscribe with Google',\n 'es': 'SuscrÃbete con Google',\n 'es-419': 'SuscrÃbete con Google',\n 'es-latam': 'SuscrÃbete con Google',\n 'es-latn': 'SuscrÃbete con Google',\n 'fr': \"S'abonner avec Google\",\n 'fr-ca': \"S'abonner avec Google\",\n 'hi': 'Google के ज़रिये सदसà¥à¤¯à¤¤à¤¾',\n 'id': 'Berlangganan dengan Google',\n 'it': 'Abbonati con Google',\n 'ja': 'Google ã§è³¼èª',\n 'ko': 'Google ì„ í†µí•œêµ¬ë…',\n 'ms': 'Langgan dengan Google',\n 'nl': 'Abonneren via Google',\n 'no': 'Abonner med Google',\n 'pl': 'Subskrybuj z Google',\n 'pt': 'Subscrever com o Google',\n 'pt-br': 'Assine com o Google',\n 'ru': 'Подпиcка через Google',\n 'sv': 'Prenumerera med Google',\n 'th': 'สมัครฟาน Google',\n 'tr': 'Google ile Abone Ol',\n 'uk': 'ПідпиÑатиÑÑ Ñ‡ÐµÑ€ÐµÐ· Google',\n 'zh-cn': '通过 Google 订阅',\n 'zh-hk': 'é€éŽ Google 訂閱',\n 'zh-tw': 'é€éŽ Google 訂閱',\n },\n 'CONTRIBUTION_TITLE_LANG_MAP': {\n 'en': 'Contribute with Google',\n 'ar': 'المساهمة باستخدام Google',\n 'de': 'Mit Google beitragen',\n 'en-au': 'Contribute with Google',\n 'en-ca': 'Contribute with Google',\n 'en-gb': 'Contribute with Google',\n 'en-us': 'Contribute with Google',\n 'es': '\tContribuye con Google',\n 'es-419': 'Contribuir con Google',\n 'es-latam': 'Contribuir con Google',\n 'es-latn': 'Contribuye con Google',\n 'fr': 'Contribuer avec Google',\n 'fr-ca': 'Contribuer avec Google',\n 'hi': 'Google खाते की मदद से योगदान करें',\n 'id': 'Berkontribusi dengan Google',\n 'it': 'Contribuisci con Google',\n 'ja': 'Google ã§å¯„付',\n 'ko': 'Googleì„ í†µí•´ 참여하기',\n 'ms': 'Sumbangkan dengan Google',\n 'nl': 'Bijdragen met Google',\n 'no': 'Bidra med Google',\n 'pl': 'Wesprzyj publikacjÄ™ przez Google',\n 'pt': 'Contribuir utilizando o Google',\n 'pt-br': 'Contribua usando o Google',\n 'ru': 'ВнеÑти ÑредÑтва через Google',\n 'sv': 'Bidra med Google',\n 'th': 'มีส่วนร่วมผ่าน Google',\n 'tr': 'Google ile Katkıda Bulun',\n 'uk': 'Зробити внеÑок через Google',\n 'zh-cn': '通过 Google æèµ ',\n 'zh-hk': 'é€éŽ Google æ供內容',\n 'zh-tw': 'é€éŽ Google æ款',\n },\n 'REGWALL_ALREADY_REGISTERED_LANG_MAP': {\n 'en': 'You have registered before.',\n 'ar': 'لقد سبق أن تسجّلت.',\n 'de': 'Du bist bereits registriert.',\n 'en-au': 'You have registered before.',\n 'en-ca': 'You have registered before.',\n 'en-gb': 'You have registered before.',\n 'en-us': 'You have registered before.',\n 'es': 'Ya te habÃas registrado anteriormente.',\n 'es-419': 'Ya te registraste antes.',\n 'fr': 'Vous vous êtes déjà inscrit.',\n 'fr-ca': 'Vous vous êtes inscrit auparavant.',\n 'hi': 'आपने पहले ही इसके लिठरजिसà¥à¤Ÿà¤° कर लिया है.',\n 'id': 'Anda telah mendaftar sebelumnya.',\n 'it': 'Registrazione già effettuata in precedenza.',\n 'ja': 'ã™ã§ã«ç™»éŒ²æ¸ˆã¿ã§ã™ã€‚',\n 'ko': 'ì´ì „ì— ë“±ë¡í•œ 사용ìžìž…니다.',\n 'ms': 'Anda telah mendaftar sebelum ini.',\n 'nl': 'Je hebt je al eerder geregistreerd.',\n 'no': 'Du er allerede registrert.',\n 'pl': 'Masz już wczeÅ›niejszÄ… rejestracjÄ™.',\n 'pt': 'Já se registou anteriormente.',\n 'pt-br': 'Você já tem um cadastro.',\n 'ru': 'Ð’Ñ‹ уже зарегиÑтрированы.',\n 'sv': 'Du har redan registrerat dig.',\n 'th': 'คุณเคยลงทะเบียนà¹à¸¥à¹‰à¸§',\n 'tr': 'Daha önce kaydolmuÅŸtunuz.',\n 'uk': 'Ви вже зареєÑтрувалиÑÑ Ñ€Ð°Ð½Ñ–ÑˆÐµ.',\n 'zh-cn': '您之å‰å·²æ³¨å†Œã€‚',\n 'zh-hk': '您之å‰å·²è¨»å†Šã€‚',\n 'zh-tw': 'ä½ å·²è¨»å†Šé€™å€‹å‡ºç‰ˆå“。',\n },\n 'NEWSLETTER_ALREADY_SIGNED_UP_LANG_MAP': {\n 'en': 'You have signed up before.',\n 'ar': 'سبق أن اشتركت ÙÙŠ النشرة الإخبارية.',\n 'de': 'Du hast dich bereits angemeldet.',\n 'en-au': 'You have signed up before.',\n 'en-ca': 'You have signed up before.',\n 'en-gb': 'You have signed up before.',\n 'en-us': 'You have signed up before.',\n 'es': 'Ya te has registrado anteriormente.',\n 'es-419': 'Ya te registraste antes.',\n 'fr': 'Vous vous êtes déjà inscrit.',\n 'fr-ca': 'Vous vous êtes inscrit auparavant.',\n 'hi': 'नà¥à¤¯à¥‚ज़लेटर के लिठपहले ही साइन अप किया जा चà¥à¤•à¤¾ है.',\n 'id': 'Anda telah mendaftar sebelumnya.',\n 'it': \"Hai già effettuato l'iscrizione.\",\n 'ja': 'ã™ã§ã«ç™»éŒ²ã•ã‚Œã¦ã„ã¾ã™ã€‚',\n 'ko': 'ì´ì „ì— ê°€ìž…í•œ 사용ìžìž…니다.',\n 'ms': 'Anda sudah mendaftar sebelum ini.',\n 'nl': 'Je hebt je al eerder aangemeld.',\n 'no': 'Du er allerede registrert.',\n 'pl': 'Już wczeÅ›niej siÄ™ zarejestrowaÅ‚eÅ›(-aÅ›).',\n 'pt': 'Já se inscreveu anteriormente.',\n 'pt-br': 'Você se inscreveu anteriormente.',\n 'ru': 'Ð’Ñ‹ уже зарегиÑтрированы.',\n 'sv': 'Du har redan registrerat dig.',\n 'th': 'คุณสมัครรับข้à¸à¸¡à¸¹à¸¥à¸¡à¸²à¸à¹ˆà¸à¸™à¹à¸¥à¹‰à¸§',\n 'tr': 'Daha önce kaydolmuÅŸtunuz.',\n 'uk': 'Ви вже зареєÑтрувалиÑÑ.',\n 'zh-cn': '您之å‰å·²æ³¨å†Œã€‚',\n 'zh-hk': '您之å‰å·²è¨‚閱。',\n 'zh-tw': 'ä½ å·²ç¶“è¨‚é–±äº†ã€‚',\n },\n 'REGWALL_REGISTER_FAILED_LANG_MAP': {\n 'en': 'Registration failed. Try registering again.',\n 'ar': 'تعذَّرت عملية التسجيل. ÙŠÙرجى إعادة المØاولة.',\n 'de': 'Registrierung fehlgeschlagen. Versuche es noch einmal.',\n 'en-au': 'Registration failed. Try registering again.',\n 'en-ca': 'Registration failed. Try registering again.',\n 'en-gb': 'Registration failed. Try registering again.',\n 'en-us': 'Registration failed. Try registering again.',\n 'es': 'No se ha podido completar el registro. Prueba a registrarte de nuevo.',\n 'es-419': 'No se pudo completar el registro. Vuelve a intentarlo.',\n 'fr': \"Échec de l'enregistrement. Réessayez.\",\n 'fr-ca': \"Échec de l'inscription. Essayez de vous inscrire à nouveau.\",\n 'hi': 'रजिसà¥à¤Ÿà¥à¤°à¥‡à¤¶à¤¨ नहीं हो सका. फिर से रजिसà¥à¤Ÿà¤° करने की कोशिश करें.',\n 'id': 'Pendaftaran gagal. Coba daftar lagi.',\n 'it': 'Registrazione non riuscita. Prova a registrarti di nuovo.',\n 'ja': '登録ã§ãã¾ã›ã‚“ã§ã—ãŸã€‚ã‚‚ã†ä¸€åº¦ãŠè©¦ã—ãã ã•ã„。',\n 'ko': '등ë¡ì— 실패했습니다. 다시 등ë¡í•´ 보세요.',\n 'ms': 'Pendaftaran gagal. Cuba mendaftar lagi.',\n 'nl': 'Registratie mislukt. Probeer opnieuw te registreren.',\n 'no': 'Registreringen mislyktes. Prøv Ã¥ registrere deg pÃ¥ nytt.',\n 'pl': 'Rejestracja siÄ™ nie udaÅ‚a. Spróbuj jeszcze raz siÄ™ zarejestrować.',\n 'pt': 'Falha no registo. Tente registar-se novamente.',\n 'pt-br': 'Não foi possÃvel fazer o registro. Tente novamente.',\n 'ru': 'Ошибка региÑтрации. Повторите попытку.',\n 'sv': 'Registreringen misslyckades. Försök att registrera dig igen.',\n 'th': 'ลงทะเบียนไม่สำเร็จ ลà¸à¸‡à¸¥à¸‡à¸—ะเบียนà¸à¸µà¸à¸„รั้ง',\n 'tr': 'Kayıt iÅŸlemi baÅŸarısız oldu. Tekrar kaydolmayı deneyin.',\n 'uk': 'Помилка реєÑтрації. Повторіть Ñпробу.',\n 'zh-cn': '注册失败。请å°è¯•é‡æ–°æ³¨å†Œã€‚',\n 'zh-hk': '註冊失敗。請嘗試é‡æ–°è¨»å†Šã€‚',\n 'zh-tw': '註冊失敗,請å†è©¦ä¸€æ¬¡ã€‚',\n },\n 'NEWSLETTER_SIGN_UP_FAILED_LANG_MAP': {\n 'en': 'Signup failed. Try signing up again.',\n 'ar': 'تعذَّرت عملية الاشتراك. ÙŠÙرجى إعادة المØاولة.',\n 'de': 'Anmeldung fehlgeschlagen. Versuche es noch einmal.',\n 'en-au': 'Sign-up failed. Try signing up again.',\n 'en-ca': 'Sign-up failed. Try signing up again.',\n 'en-gb': 'Sign-up failed. Try signing up again.',\n 'en-us': 'Sign-up failed. Try signing up again.',\n 'es': 'No se ha podido completar la suscripción. Prueba a suscribirte de nuevo.',\n 'es-419': 'Se produjo un error de registro. Vuelve a intentarlo.',\n 'fr': \"Échec de l'inscription. Réessayez.\",\n 'fr-ca': \"Échec de l'inscription. Essayez de vous inscrire à nouveau.\",\n 'hi': 'साइन अप नहीं किया जा सका. फिर से साइन अप करने की कोशिश करें.',\n 'id': 'Pendaftaran gagal. Coba daftar lagi.',\n 'it': 'Iscrizione non riuscita. Prova a iscriverti di nuovo.',\n 'ja': '登録ã§ãã¾ã›ã‚“ã§ã—ãŸã€‚ã‚‚ã†ä¸€åº¦ãŠè©¦ã—ãã ã•ã„。',\n 'ko': 'ê°€ìž…ì— ì‹¤íŒ¨í–ˆìŠµë‹ˆë‹¤. 다시 가입해 보세요.',\n 'ms': 'Daftar gagal. Cuba daftar lagi.',\n 'nl': 'Aanmelding mislukt. Probeer opnieuw aan te melden.',\n 'no': 'Registreringen mislyktes. Prøv Ã¥ registrere deg pÃ¥ nytt.',\n 'pl': 'Rejestracja siÄ™ nie udaÅ‚a. Spróbuj jeszcze raz siÄ™ zarejestrować.',\n 'pt': 'Falha na inscrição. Tente inscrever-se novamente.',\n 'pt-br': 'Não foi possÃvel se inscrever. Tente novamente.',\n 'ru': 'Ðе удалоÑÑŒ зарегиÑтрироватьÑÑ. Повторите попытку.',\n 'sv': 'Registreringen misslyckades. Försök att registrera dig igen.',\n 'th': 'ลงชื่à¸à¸ªà¸¡à¸±à¸„รใช้ไม่สำเร็จ ลà¸à¸‡à¸¥à¸‡à¸Šà¸·à¹ˆà¸à¸ªà¸¡à¸±à¸„รใช้à¸à¸µà¸à¸„รั้ง',\n 'tr': 'Kaydolma iÅŸlemi baÅŸarısız oldu. Tekrar kaydolmayı deneyin.',\n 'uk': 'Помилка реєÑтрації. Повторіть Ñпробу.',\n 'zh-cn': '注册失败。请å°è¯•é‡æ–°æ³¨å†Œã€‚',\n 'zh-hk': '申請失敗。請嘗試é‡æ–°ç”³è«‹ã€‚',\n 'zh-tw': '訂閱失敗,請å†è©¦ä¸€æ¬¡ã€‚',\n },\n 'REGWALL_ACCOUNT_CREATED_LANG_MAP': {\n 'en': 'Created an account with
user@gmail.com%s',\n 'ar': 'تم إنشاء Øساب باستخدام
user@gmail.com%s.',\n 'de': 'Konto bei
user@gmail.com%s wurde erstellt',\n 'en-au':\n 'Created an account with
user@gmail.com%s',\n 'en-ca':\n 'Created an account with
user@gmail.com%s',\n 'en-gb':\n 'Created an account with
user@gmail.com%s',\n 'en-us':\n 'Created an account with
user@gmail.com%s',\n 'es': 'Has creado una cuenta con
user@gmail.com%s',\n 'es-419':\n 'Se creó una cuenta con
user@gmail.com%s',\n 'fr': 'A créé un compte avec
user@gmail.com%s',\n 'fr-ca':\n 'Un compte a été créé avec l\\'adresse
user@gmail.com%s',\n 'hi': '
user@gmail.com%s का इसà¥à¤¤à¥‡à¤®à¤¾à¤² करके, à¤à¤• खाता बनाया गया',\n 'id': 'Membuat akun dengan
user@gmail.com%s',\n 'it': 'È stato creato un account con l\\'indirizzo
user@gmail.com%s',\n 'ja': '
user@gmail.com%s ã§ã‚¢ã‚«ã‚¦ãƒ³ãƒˆã‚’作æˆã—ã¾ã—ãŸ',\n 'ko': '
user@gmail.com%s(으)ë¡œ ê³„ì •ì„ ë§Œë“¤ì—ˆìŠµë‹ˆë‹¤.',\n 'ms': 'Membuat akaun dengan
user@gmail.com%s',\n 'nl': 'Account gemaakt met
user@gmail.com%s',\n 'no': 'Du har opprettet en konto med
user@gmail.com%s',\n 'pl': 'Utworzono konto za pomocÄ… adresu
user@gmail.com%s',\n 'pt': 'Criou uma conta com
user@gmail.com%s',\n 'pt-br':\n 'Conta criada com o e-mail
user@gmail.com%s',\n 'ru': 'Ð’Ñ‹ зарегиÑтрировали аккаунт на адреÑ
user@gmail.com%s.',\n 'sv': 'Du skapade ett konto med
user@gmail.com%s',\n 'th': 'สร้างบัà¸à¸Šà¸µà¸”้วย
user@gmail.com%s',\n 'tr': '
user@gmail.com%s ile bir hesap oluÅŸturun',\n 'uk': 'Обліковий Ð·Ð°Ð¿Ð¸Ñ Ñтворено за допомогою електронної адреÑи
user@gmail.com%s',\n 'zh-cn': '已使用
user@gmail.com%s 创建å¸å·',\n 'zh-hk': '已使用
user@gmail.com%s 建立帳戶',\n 'zh-tw': '已使用
user@gmail.com%s 建立帳戶',\n },\n 'NEWSLETTER_SIGNED_UP_LANG_MAP': {\n 'en': 'Signed up with
user@gmail.com%s for the newsletter',\n 'ar': 'تم الاشتراك ÙÙŠ النشرة الإخبارية باستخدام
user@gmail.com%s.',\n 'de': 'Du hast dich für den Newsletter von
user@gmail.com%s angemeldet',\n 'en-au':\n 'Signed up with
user@gmail.com%s for the newsletter',\n 'en-ca':\n 'Signed up with
user@gmail.com%s for the newsletter',\n 'en-gb':\n 'Signed up with
user@gmail.com%s for the newsletter',\n 'en-us':\n 'Signed up with
user@gmail.com%s for the newsletter',\n 'es': 'Te has suscrito a la newsletter con
user@gmail.com%s',\n 'es-419':\n 'Te registraste con
user@gmail.com%s para recibir el boletÃn informativo',\n 'fr': 'S\\'est abonné à la newsletter avec
user@gmail.com%s',\n 'fr-ca':\n 'Vous êtes inscrit au bulletin d\\'information avec l\\'adresse
user@gmail.com%s',\n 'hi': 'नà¥à¤¯à¥‚ज़लेटर पाने के लिà¤,
user@gmail.com%s से साइन अप किया गया',\n 'id': 'Mendaftar dengan
user@gmail.com%s untuk mendapatkan newsletter',\n 'it': 'Iscrizione alla newsletter con l\\'indirizzo
user@gmail.com%s effettuata',\n 'ja': '
user@gmail.com%s ã§ãƒ‹ãƒ¥ãƒ¼ã‚¹ãƒ¬ã‚¿ãƒ¼ã‚’登録ã—ã¾ã—ãŸ',\n 'ko': '
user@gmail.com%s(으)ë¡œ ë‰´ìŠ¤ë ˆí„°ì— ê°€ìž…í–ˆìŠµë‹ˆë‹¤.',\n 'ms': 'Mendaftar dengan
user@gmail.com%s untuk surat berita',\n 'nl': 'Aangemeld met
user@gmail.com%s voor de nieuwsbrief',\n 'no': 'Du har registrert deg for nyhetsbrevet med
user@gmail.com%s',\n 'pl': 'Zapisano siÄ™ na newsletter za pomocÄ… adresu
user@gmail.com%s',\n 'pt': 'Inscreveu-se com
user@gmail.com%s no boletim informativo',\n 'pt-br':\n 'Inscrição na newsletter feita com o e-mail
user@gmail.com%s',\n 'ru': 'Ð’Ñ‹ подпиÑалиÑÑŒ на новоÑтную раÑÑылку, иÑÐ¿Ð¾Ð»ÑŒÐ·ÑƒÑ Ð°ÐºÐºÐ°ÑƒÐ½Ñ‚
user@gmail.com%s.',\n 'sv': 'Du registrerade dig för nyhetsbrevet med
user@gmail.com%s',\n 'th': 'ลงชื่à¸à¹€à¸‚้าใช้ด้วย
user@gmail.com%s สำหรับจดหมายข่าว',\n 'tr': 'Bülten için
user@gmail.com%s ile kaydoldunuz',\n 'uk': 'Ви підпиÑалиÑÑ Ð½Ð° інформаційні лиÑти на електронну адреÑу
user@gmail.com%s',\n 'zh-cn': '已使用
user@gmail.com%s 订阅简报',\n 'zh-hk': '已使用
user@gmail.com%s 註冊通訊',\n 'zh-tw':\n '已使用
user@gmail.com%s 訂閱電åå ±',\n },\n 'NO_MEMBERSHIP_FOUND_LANG_MAP': {\n 'en': 'No membership found',\n 'ar': 'لم يتم العثور على أي اشتراك.',\n 'de': 'Keine Mitgliedschaftsdaten gefunden',\n 'en-au': 'No membership found',\n 'en-ca': 'No membership found',\n 'en-gb': 'No membership found',\n 'en-us': 'No membership found',\n 'es': 'No se han encontrado suscripciones',\n 'es-419': 'No se encontró ninguna membresÃa',\n 'fr': 'Aucun abonnement trouvé',\n 'fr-ca': 'Aucun abonnement trouvé',\n 'hi': 'पैसे चà¥à¤•à¤¾à¤•à¤° ली जाने वाली कोई सदसà¥à¤¯à¤¤à¤¾ नहीं मिली',\n 'id': 'Langganan tidak ditemukan',\n 'it': 'Nessun abbonamento trovato',\n 'ja': 'メンãƒãƒ¼ã‚·ãƒƒãƒ—ãŒè¦‹ã¤ã‹ã‚Šã¾ã›ã‚“',\n 'ko': 'ë©¤ë²„ì‹ ì •ë³´ë¥¼ ì°¾ì„ ìˆ˜ 없습니다.',\n 'ms': 'Tiada keahlian ditemukan',\n 'nl': 'Geen lidmaatschap gevonden',\n 'no': 'Fant ingen abonnementer',\n 'pl': 'Nie znaleziono subskrypcji',\n 'pt': 'Nenhuma subscrição encontrada',\n 'pt-br': 'Nenhuma assinatura foi encontrada',\n 'ru': 'ПодпиÑка не найдена.',\n 'sv': 'Inget medlemskap hittades',\n 'th': 'ไม่พบà¸à¸²à¸£à¹€à¸›à¹‡à¸™à¸ªà¸¡à¸²à¸Šà¸´à¸',\n 'tr': 'Ãœyelik bulunamadı',\n 'uk': 'Ðемає підпиÑок',\n 'zh-cn': '未找到会员资料',\n 'zh-hk': '找ä¸åˆ°æœƒç±',\n 'zh-tw': '找ä¸åˆ°æœƒå“¡è³‡æ–™',\n },\n 'CLOSE_BUTTON_DESCRIPTION': {\n 'en': 'Close dialog',\n 'ar': 'إغلاق مربّع الØوار',\n 'de': 'Dialogfeld schließen',\n 'en-au': 'Close dialog',\n 'en-ca': 'Close dialog',\n 'en-gb': 'Close dialog',\n 'en-us': 'Close dialog',\n 'es': 'Cerrar cuadro de diálogo',\n 'es-419': 'Cerrar diálogo',\n 'fr': 'Fermer la boîte de dialogue',\n 'fr-ca': 'Fermer la boîte de dialogue',\n 'hi': 'डायलॉग बॉकà¥à¤¸ बंद करें',\n 'id': 'Tutup dialog',\n 'it': 'Chiudi la finestra di dialogo',\n 'ja': 'ダイアãƒã‚°ã‚’é–‰ã˜ã‚‹',\n 'ko': '대화ìƒìž 닫기',\n 'ms': 'Tutup dialog',\n 'nl': 'Dialoogvenster sluiten',\n 'no': 'Lukk dialogboksen',\n 'pl': 'Zamknij okno',\n 'pt': 'Fechar caixa de diálogo',\n 'pt-br': 'Fechar caixa de diálogo',\n 'ru': 'Закрыть диалоговое окно',\n 'sv': 'Stäng dialogrutan',\n 'th': 'ปิดà¸à¸¥à¹ˆà¸à¸‡à¹‚ต้ตà¸à¸š',\n 'tr': 'Ä°letiÅŸim kutusunu kapat',\n 'uk': 'Закрити вікно',\n 'zh-cn': 'å…³é—对è¯æ¡†',\n 'zh-hk': 'é–‚å°è©±æ¡†',\n 'zh-tw': '關閉å°è©±æ–¹å¡Š',\n },\n 'CONTRIBUTE': {\n 'en': 'Contribute',\n 'ar': 'مساهمة',\n 'de': 'Beitragen',\n 'en-au': 'Contribute',\n 'en-ca': 'Contribute',\n 'en-gb': 'Contribute',\n 'en-us': 'Contribute',\n 'es': 'Contribuir',\n 'es-419': 'Contribuir',\n 'fr': 'Contribuer',\n 'fr-ca': 'Faire une contribution',\n 'hi': 'योगदान दें',\n 'id': 'Beri kontribusi',\n 'it': 'Contribuisci',\n 'ja': '寄付',\n 'ko': '후ì›',\n 'ms': 'Sumbang',\n 'nl': 'Bijdragen',\n 'no': 'Bidra',\n 'pl': 'Przekaż darowiznÄ™',\n 'pt': 'Contribuir',\n 'pt-br': 'Contribuir',\n 'ru': 'Сделать взноÑ',\n 'sv': 'Bidra',\n 'th': 'สนับสนุน',\n 'tr': 'Katkıda bulun',\n 'uk': 'Зробити внеÑок',\n 'zh-cn': 'æèµ ',\n 'zh-hk': '資助',\n 'zh-tw': 'æ款',\n },\n 'SUBSCRIBE': {\n 'en': 'Subscribe',\n 'ar': 'اشتراك',\n 'de': 'Abonnieren',\n 'en-au': 'Subscribe',\n 'en-ca': 'Subscribe',\n 'en-gb': 'Subscribe',\n 'en-us': 'Subscribe',\n 'es': 'Suscribirme',\n 'es-419': 'Suscribirse',\n 'fr': \"S'abonner\",\n 'fr-ca': \"S'abonner\",\n 'hi': 'सदसà¥à¤¯à¤¤à¤¾ लें',\n 'id': 'Langganan',\n 'it': 'Iscriviti',\n 'ja': 'è³¼èª',\n 'ko': '구ë…',\n 'ms': 'Langgan',\n 'nl': 'Abonneren',\n 'no': 'Abonner',\n 'pl': 'Subskrybuj',\n 'pt': 'Subscrever',\n 'pt-br': 'Fazer inscrição',\n 'ru': 'ПодпиÑатьÑÑ',\n 'sv': 'Prenumerera',\n 'th': 'สมัครสมาชิà¸',\n 'tr': 'Abone ol',\n 'uk': 'ПідпиÑатиÑÑ',\n 'zh-cn': '订阅',\n 'zh-hk': '訂閱',\n 'zh-tw': '訂閱',\n },\n 'ALREADY_A_CONTRIBUTOR': {\n 'en': 'Already a contributor?',\n 'ar': 'هل أنت مساهم Øالي؟',\n 'de': 'Du bist schon Beitragende/r?',\n 'en-au': 'Already a contributor?',\n 'en-ca': 'Already a contributor?',\n 'en-gb': 'Already a contributor?',\n 'en-us': 'Already a contributor?',\n 'es': '¿Ya has contribuido?',\n 'es-419': '¿Ya contribuyes?',\n 'fr': 'Déjà contributeur ?',\n 'fr-ca': 'Vous êtes déjà contributeur?',\n 'hi': 'कà¥à¤¯à¤¾ आपने पहले योगदान दिया है?',\n 'id': 'Sudah menjadi kontributor?',\n 'it': 'Hai già dato contributi?',\n 'ja': 'ã™ã§ã«å¯„付ã—ã¦ã„ã¾ã™ã‹ï¼Ÿ',\n 'ko': 'ì´ë¯¸ 후ì›í•˜ê³ ê³„ì‹ ê°€ìš”?',\n 'ms': 'Sudah menjadi penyumbang?',\n 'nl': 'Ben je al een bijdrager?',\n 'no': 'Har du allerede bidratt?',\n 'pl': 'Już przekazujesz darowizny?',\n 'pt': 'Já contribui?',\n 'pt-br': 'Já faz contribuições?',\n 'ru': 'Уже делаете взноÑÑ‹?',\n 'sv': 'Är du redan en bidragsgivare?',\n 'th': 'หาà¸à¹€à¸›à¹‡à¸™à¸œà¸¹à¹‰à¸ªà¸™à¸±à¸šà¸ªà¸™à¸¸à¸™à¸à¸¢à¸¹à¹ˆà¹à¸¥à¹‰à¸§',\n 'tr': 'Halihazırda katkıda bulunuyor musunuz?',\n 'uk': 'Уже робите внеÑки?',\n 'zh-cn': '已是æèµ è€…ï¼Ÿ',\n 'zh-hk': '已是資助者?',\n 'zh-tw': '已經是æ款者了嗎?',\n },\n 'ALREADY_A_SUBSCRIBER': {\n 'en': 'Already a subscriber?',\n 'ar': 'هل أنت مشترك Øالي؟',\n 'de': 'Du bist bereits Abonnent/in?',\n 'en-au': 'Already a subscriber?',\n 'en-ca': 'Already a subscriber?',\n 'en-gb': 'Already a subscriber?',\n 'en-us': 'Already a subscriber?',\n 'es': '¿Ya te has suscrito?',\n 'es-419': '¿Ya te suscribiste?',\n 'fr': 'Déjà abonné ?',\n 'fr-ca': 'Vous êtes déjà abonné?',\n 'hi': 'कà¥à¤¯à¤¾ आपने पहले ही सदसà¥à¤¯à¤¤à¤¾ ले ली है?',\n 'id': 'Sudah berlangganan?',\n 'it': \"Hai già l'abbonamento?\",\n 'ja': 'è³¼èªæ¸ˆã¿ã§ã™ã‹ï¼Ÿ',\n 'ko': 'ì´ë¯¸ êµ¬ë… ì¤‘ì´ì‹ 가요?',\n 'ms': 'Sudah menjadi pelanggan?',\n 'nl': 'Ben je al abonnee?',\n 'no': 'Er du allerede abonnent?',\n 'pl': 'Już subskrybujesz?',\n 'pt': 'Já subscreve?',\n 'pt-br': 'Já é assinante?',\n 'ru': 'Уже подпиÑаны?',\n 'sv': 'Prenumererar du redan?',\n 'th': 'หาà¸à¹€à¸›à¹‡à¸™à¸ªà¸¡à¸²à¸Šà¸´à¸à¸à¸¢à¸¹à¹ˆà¹à¸¥à¹‰à¸§',\n 'tr': 'Halihazırda abone misiniz?',\n 'uk': 'Уже підпиÑалиÑÑ?',\n 'zh-cn': '已订阅?',\n 'zh-hk': '已經是訂閱者?',\n 'zh-tw': '已經是訂閱者了嗎?',\n },\n 'THANKS_FOR_VIEWING_THIS_AD': {\n 'en': 'Thanks for viewing this ad',\n 'ar': 'نشكرك على مشاهدة هذا الإعلان.',\n 'de': 'Vielen Dank, dass du dir diese Anzeige angesehen hast',\n 'en-au': 'Thanks for viewing this ad',\n 'en-ca': 'Thanks for viewing this ad',\n 'en-gb': 'Thanks for viewing this ad',\n 'en-us': 'Thanks for viewing this ad',\n 'es': 'Gracias por ver este anuncio',\n 'es-419': 'Gracias por ver este anuncio',\n 'fr': \"Merci d'avoir regardé cette annonce\",\n 'fr-ca': \"Merci d'avoir regardé cette annonce\",\n 'hi': 'यह विजà¥à¤žà¤¾à¤ªà¤¨ देखने के लिठधनà¥à¤¯à¤µà¤¾à¤¦',\n 'id': 'Terima kasih telah melihat iklan ini',\n 'it': 'Grazie per aver visualizzato questo annuncio',\n 'ja': 'ã“ã®åºƒå‘Šã‚’ã”覧ã„ãŸã ãã‚ã‚ŠãŒã¨ã†ã”ã–ã„ã¾ã™',\n 'ko': 'ì´ ê´‘ê³ ë¥¼ ì‹œì²í•´ 주셔서 ê°ì‚¬í•©ë‹ˆë‹¤.',\n 'ms': 'Terima kasih kerana menonton iklan ini',\n 'nl': 'Bedankt dat je deze advertentie hebt bekeken',\n 'no': 'Takk for at du sÃ¥ pÃ¥ denne annonsen',\n 'pl': 'DziÄ™kujemy za obejrzenie tej reklamy',\n 'pt': 'Obrigado por ver este anúncio',\n 'pt-br': 'Agradecemos por assistir este anúncio',\n 'ru': 'СпаÑибо, что поÑмотрели Ñту рекламу!',\n 'sv': 'Tack för att du tittade pÃ¥ den här annonsen',\n 'th': 'ขà¸à¸šà¸„ุณที่ดูโฆษณานี้',\n 'tr': 'Bu reklamı görüntülediÄŸiniz için teÅŸekkür ederiz',\n 'uk': 'ДÑкуємо, що переглÑнули це оголошеннÑ',\n 'zh-cn': '感谢观看æ¤å¹¿å‘Š',\n 'zh-hk': 'æ„Ÿè¬è§€çœ‹æ¤å»£å‘Š',\n 'zh-tw': 'æ„Ÿè¬è§€çœ‹é€™å‰‡å»£å‘Š',\n },\n 'VIEW_AN_AD': {\n 'en': 'View an ad',\n 'ar': 'عرض إعلان',\n 'de': 'Anzeige ansehen',\n 'en-au': 'View an ad',\n 'en-ca': 'View an ad',\n 'en-gb': 'View an ad',\n 'en-us': 'View an ad',\n 'es': 'Ver un anuncio',\n 'es-419': 'Ver un anuncio',\n 'fr': 'Visionner une annonce',\n 'fr-ca': 'Afficher une annonce',\n 'hi': 'कोई विजà¥à¤žà¤¾à¤ªà¤¨ देखें',\n 'id': 'Lihat iklan',\n 'it': 'Visualizza un annuncio',\n 'ja': '広告を表示',\n 'ko': 'ê´‘ê³ ë³´ê¸°',\n 'ms': 'Lihat iklan',\n 'nl': 'Een advertentie bekijken',\n 'no': 'Se en annonsv',\n 'pl': 'Obejrzyj reklamÄ™',\n 'pt': 'Mostrar um anúncio',\n 'pt-br': 'Ver anúncio',\n 'ru': 'ПоÑмотреть рекламу',\n 'sv': 'Se en annons',\n 'th': 'ดูโฆษณา',\n 'tr': 'Reklam izle',\n 'uk': 'ПереглÑнути оголошеннÑ',\n 'zh-cn': '观看广告',\n 'zh-hk': '觀看廣告',\n 'zh-tw': '觀看廣告',\n },\n // Message ID: 7478828886861577969\n 'BACK_TO_HOMEPAGE': {\n 'en': 'Back to homepage',\n 'ar': 'الرجوع إلى الصÙØØ© الرئيسية',\n 'de': 'Zurück zur Startseite',\n 'en-au': 'Back to homepage',\n 'en-ca': 'Back to homepage',\n 'en-gb': 'Back to homepage',\n 'en-us': 'Back to homepage',\n 'es': 'Volver a la página principal',\n 'es-419': 'Volver a la página principal',\n 'fr': \"Retourner à la page d'accueil\",\n 'fr-ca': \"Retour à la page d'accueil\",\n 'hi': 'होम पेज पर वापस जाà¤à¤‚',\n 'id': 'Kembali ke halaman beranda',\n 'it': 'Torna alla home page',\n 'ja': 'ホームページã«æˆ»ã‚‹',\n 'ko': '홈페ì´ì§€ë¡œ ëŒì•„가기',\n 'ms': 'Kembali kepada halaman utama',\n 'nl': 'Terug naar homepage',\n 'no': 'Tilbake til startsiden',\n 'pl': 'Powrót do strony głównej',\n 'pt': 'Volte à página inicial',\n 'pt-br': 'Voltar à página inicial',\n 'ru': 'ВернутьÑÑ Ð½Ð° главную Ñтраницу',\n 'sv': 'Tillbaka till startsidan',\n 'th': 'à¸à¸¥à¸±à¸šà¹„ปที่หน้าà¹à¸£à¸',\n 'tr': 'Ana sayfaya geri dön',\n 'uk': 'Ðазад на головну Ñторінку',\n 'zh-cn': '返回首页',\n 'zh-hk': '返回首é ',\n 'zh-tw': '返回首é ',\n },\n};\n","/**\n * Copyright 2021 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst Constants = {\n /**\n * IAB Audience taxonomy version for logging PPS values to localStorage.\n * Value mapped to googletag.enums.Taxonomy.IAB_AUDIENCE_1_1.\n */\n PPS_AUDIENCE_TAXONOMY_KEY: 1,\n};\n\nconst StorageKeys = {\n /**\n * Local storage key for swgUserToken.\n */\n USER_TOKEN: 'USER_TOKEN',\n\n /**\n * Local storage key for read time.\n */\n READ_TIME: 'READ_TIME',\n\n /**\n * Local storage key for cacheable entitlements.\n */\n ENTITLEMENTS: 'ents',\n\n /**\n * Local storage key for whether credential isReadyToPay.\n */\n IS_READY_TO_PAY: 'isreadytopay',\n\n /**\n * Local storage key for redirect.\n */\n REDIRECT: 'subscribe.google.com:rk',\n\n /**\n * Local storage key for survey completed timestamps.\n */\n SURVEY_COMPLETED: 'surveycompleted',\n\n /**\n * Local storage key for survey data transfer failure timestamps.\n */\n SURVEY_DATA_TRANSFER_FAILED: 'surveydatatransferfailed',\n\n /**\n * Local storage key for whether toast was shown.\n */\n TOAST: 'toast',\n\n /**\n * Local storage key for frequency capping timestamps.\n */\n TIMESTAMPS: 'tsp',\n};\n\nconst StorageKeysWithoutPublicationIdSuffix = {\n /**\n * Local storage key for IAB Audience Taxonomy values. It must take on the\n * 'values' as defined by the PPS GPT API.\n * In order to maintain legacy code snippet that's provided to publishers without publicationId suffix, the stroage key should remain as \"subscribe.google.com:ppstaxonomies\".\n */\n PPS_TAXONOMIES: 'ppstaxonomies',\n};\n\nexport {Constants, StorageKeys, StorageKeysWithoutPublicationIdSuffix};\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {Doc} from '../model/doc';\n\nexport const styleType = 'text/css';\n\n/**\n * Add attributes to an element.\n */\nfunction addAttributesToElement(\n element: Element,\n attributes: {[key: string]: string} = {}\n) {\n for (const [key, value] of Object.entries(attributes)) {\n element.setAttribute(key, value);\n }\n}\n\n/**\n * Create a new element on document with specified tagName and attributes.\n */\nexport function createElement
(\n doc: Document,\n tagName: string,\n attributes: {[key: string]: string},\n content?: string\n): T {\n const element = doc.createElement(tagName) as T;\n\n addAttributesToElement(element, attributes);\n\n if (content) {\n element.textContent = content;\n }\n\n return element;\n}\n\n/**\n * Removes the element.\n */\nexport function removeElement(element: Element) {\n if (element.parentElement) {\n element.parentElement.removeChild(element);\n }\n}\n\n/**\n * Removes all children from the parent element.\n */\nexport function removeChildren(parent: Element) {\n parent.textContent = '';\n}\n\n/**\n * Injects the provided styles in the HEAD section of the document.\n * @param doc The document object.\n * @param styleText The style string.\n */\nexport function injectStyleSheet(doc: Doc, styleText: string): Element {\n const styleElement: HTMLStyleElement = createElement(\n doc.getWin().document,\n 'style',\n {\n 'type': styleType,\n }\n );\n styleElement.textContent = styleText;\n doc.getHead()?.appendChild(styleElement);\n return styleElement;\n}\n\n/**\n * Whether the node has a next node in the document order.\n * This means either:\n * a. The node itself has a nextSibling.\n * b. Any of the node ancestors has a nextSibling.\n */\nexport function hasNextNodeInDocumentOrder(node: Node): boolean {\n let currentNode: Node | null = node;\n do {\n if (currentNode.nextSibling) {\n return true;\n }\n } while ((currentNode = currentNode.parentNode));\n return false;\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Default styles to be set for top level friendly iframe.\n * Some attributes are not included such as height, left, margin-left; since\n * these attributes are updated by @media queries and having these values\n * defined here as !important does not work on IE/edge browsers.\n */\nexport const defaultStyles = {\n 'align-content': 'normal',\n 'animation': 'none',\n 'align-items': 'normal',\n 'align-self': 'auto',\n 'alignment-baseline': 'auto',\n 'backface-visibility': 'hidden',\n 'background-clip': 'border-box',\n 'background-image': 'none',\n 'baseline-shift': '0',\n 'block-size': 'auto',\n 'border': 'none',\n 'border-collapse': 'separate',\n 'bottom': '0',\n 'box-sizing': 'border-box',\n 'break-after': 'auto',\n 'break-before': 'auto',\n 'break-inside': 'auto',\n 'buffered-rendering': 'auto',\n 'caption-side': 'top',\n 'caret-color': 'rgb(51, 51, 51)',\n 'clear': 'none',\n 'color': 'rgb(51, 51, 51)',\n 'color-rendering': 'auto',\n 'column-count': 'auto',\n 'column-fill': 'balance',\n 'column-gap': 'normal',\n 'column-rule-color': 'rgb(51, 51, 51)',\n 'column-rule-style': 'none',\n 'column-rule-width': '0',\n 'column-span': 'none',\n 'column-width': 'auto',\n 'contain': 'none',\n 'counter-increment': 'none',\n 'counter-reset': 'none',\n 'cursor': 'auto',\n 'direction': 'inherit',\n 'display': 'block',\n 'empty-cells': 'show',\n 'filter': 'none',\n 'flex': 'none', // flex-grow, flex-shrink, and flex-basis.\n 'flex-flow': 'row nowrap', // flex-direction, flex-wrap.\n 'float': 'none',\n 'flood-color': 'rgb(0, 0, 0)',\n 'flood-opacity': '1',\n 'font': 'none',\n 'font-size': 'medium',\n 'font-family': '',\n 'height': 'auto',\n 'hyphens': 'manual',\n 'image-rendering': 'auto',\n 'inline-size': '', // Setting to 'auto' will not allow override.\n 'isolation': 'auto',\n 'justify-content': 'normal',\n 'justify-items': 'normal',\n 'justify-self': 'auto',\n 'letter-spacing': 'normal',\n 'lighting-color': 'rgb(255, 255, 255)',\n 'line-break': 'auto',\n 'line-height': 'normal',\n 'margin-bottom': '0',\n 'mask': 'none',\n 'max-block-size': 'none',\n 'max-height': 'none',\n 'max-inline-size': 'none',\n 'max-width': 'none',\n 'min-block-size': 'none',\n 'min-height': '0',\n 'min-inline-size': '0',\n 'min-width': '0',\n 'mix-blend-mode': 'normal',\n 'object-fit': 'fill', // Important for Safari browser.\n 'offset-distance': 'none', // Chrome only (Experimental).\n 'offset-path': 'none', // Chrome only (Experimental).\n 'offset-rotate': 'auto 0deg', // Chrome only (Experimental).\n 'opacity': '1',\n 'order': '0',\n 'orphans': '2',\n 'outline': 'none',\n 'overflow-anchor': 'auto',\n 'overflow-wrap': 'normal',\n 'overflow': 'visible',\n 'padding': '0',\n 'page': '',\n 'perspective': 'none',\n 'pointer-events': 'auto',\n 'position': 'static',\n 'quotes': '',\n 'resize': 'none',\n 'right': '0',\n 'scroll-behavior': 'auto',\n 'tab-size': '8', // Only Chrome, Safari (Experimental).\n 'table-layout': 'auto',\n 'text-align': 'start',\n 'text-align-last': 'auto',\n 'text-anchor': 'start',\n 'text-combine-upright': 'none',\n 'text-decoration': 'none',\n 'text-indent': '0',\n 'text-orientation': 'mixed',\n 'text-overflow': 'clip',\n 'text-rendering': 'auto',\n 'text-shadow': 'none',\n 'text-size-adjust': 'auto',\n 'text-transform': 'none',\n 'text-underline-position': 'auto',\n 'top': 'auto',\n 'touch-action': 'auto',\n 'transform': 'none',\n 'transition': 'none 0s ease 0s',\n 'unicode-bidi': 'normal',\n 'user-select': 'auto',\n 'vector-effect': 'none',\n 'vertical-align': 'baseline',\n 'visibility': 'visible',\n 'white-space': 'normal',\n 'widows': '2',\n 'word-break': 'normal',\n 'word-spacing': '0',\n 'word-wrap': 'normal',\n 'writing-mode': 'horizontal-tb',\n 'zoom': '1',\n 'z-index': 'auto',\n};\n\n/**\n * Sets the CSS styles of the specified element with !important. The styles\n * are specified as a map from CSS property names to their values.\n */\nexport function setImportantStyles(\n element: HTMLElement,\n styles: {[property: string]: string}\n) {\n for (const [property, value] of Object.entries(styles)) {\n element.style.setProperty(property, value.toString(), 'important');\n }\n}\n\n/**\n * Sets the CSS style of the specified element.\n */\nexport function setStyle(\n element: HTMLElement,\n property: string,\n value: string\n) {\n element.style.setProperty(property, value);\n}\n\n/**\n * Returns the value of the CSS style of the specified element.\n */\nexport function getStyle(element: HTMLElement, property: string): string {\n return element.style.getPropertyValue(property);\n}\n\n/**\n * Sets the CSS styles of the specified element. The styles\n * a specified as a map from CSS property names to their values.\n */\nexport function setStyles(\n element: HTMLElement,\n styles: {[property: string]: string}\n) {\n for (const [property, value] of Object.entries(styles)) {\n setStyle(element, property, value);\n }\n}\n\n/**\n * Resets styles that were set dynamically (i.e. inline)\n */\nexport function resetStyles(element: HTMLElement, properties: Array) {\n for (const property of properties) {\n setStyle(element, property, '');\n }\n}\n\n/**\n * Resets all the styles of an element to a given value. Defaults to null.\n * The valid values are 'inherit', 'initial', 'unset' or null.\n */\nexport function resetAllStyles(element: HTMLElement) {\n setImportantStyles(element, defaultStyles);\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {setImportantStyles} from './style';\n\n/**\n * Returns a promise which is resolved after the given duration of animation\n * @param el - Element to be observed.\n * @param props - properties to be animated.\n * @param durationMillis - duration of animation.\n * @param curve - transition function for the animation.\n * @return Promise which resolves once the animation is done playing.\n */\nexport async function transition(\n el: HTMLElement,\n props: {[key: string]: string},\n durationMillis: number,\n curve: string\n): Promise {\n const win = el.ownerDocument.defaultView!;\n const previousTransitionValue = el.style.transition || '';\n\n await new Promise((resolve) => {\n win.setTimeout(() => {\n win.setTimeout(resolve, durationMillis);\n const tr = `${durationMillis}ms ${curve}`;\n setImportantStyles(\n el,\n Object.assign(\n {\n 'transition': `transform ${tr}, opacity ${tr}`,\n },\n props\n )\n );\n });\n });\n\n setImportantStyles(el, {\n 'transition': previousTransitionValue,\n });\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {ActivityPorts} from '../components/activities';\nimport {Deps} from '../runtime/deps';\nimport {Doc} from '../model/doc';\nimport {createElement} from '../utils/dom';\nimport {resetStyles, setImportantStyles} from '../utils/style';\nimport {transition} from '../utils/animation';\n\nexport const toastImportantStyles = {\n 'height': '0',\n};\n\nexport interface ToastSpecDef {\n text: string;\n action?: {label: string; handler: () => void};\n}\n\nconst iframeAttributes = {\n 'frameborder': '0',\n 'scrolling': 'no',\n 'class': 'swg-toast',\n};\n\n/**\n * The class Notification toast.\n */\nexport class Toast {\n private readonly doc_: Doc;\n private readonly activityPorts_: ActivityPorts;\n private animating_: Promise | null = null;\n private readonly iframe_: HTMLIFrameElement;\n\n constructor(\n deps: Deps,\n private readonly src_: string,\n private readonly args_: {[key: string]: string} = {}\n ) {\n this.doc_ = deps.doc();\n\n this.activityPorts_ = deps.activities();\n\n this.iframe_ = createElement(\n this.doc_.getWin().document,\n 'iframe',\n iframeAttributes\n );\n\n setImportantStyles(this.iframe_, toastImportantStyles);\n }\n\n /**\n * Returns the iframe element.\n */\n getElement(): HTMLIFrameElement {\n return this.iframe_;\n }\n\n /**\n * Opens the notification toast.\n */\n open(): Promise {\n this.doc_.getBody()?.appendChild(this.iframe_); // Fires onload.\n return this.buildToast_();\n }\n\n /**\n * Builds the content of the iframe. On load, animates the toast.\n */\n private async buildToast_(): Promise {\n const toastDurationSeconds = 7;\n const port = await this.activityPorts_.openIframe(\n this.iframe_,\n this.src_,\n this.args_\n );\n await port.whenReady();\n resetStyles(this.iframe_, ['height']);\n\n this.animating_ = this.animate_({\n callback: () => {\n setImportantStyles(this.iframe_, {\n 'transform': 'translateY(100%)',\n 'opacity': '1',\n 'visibility': 'visible',\n });\n return transition(\n this.iframe_,\n {\n 'transform': 'translateY(0)',\n 'opacity': '1',\n 'visibility': 'visible',\n },\n 400,\n 'ease-out'\n );\n },\n });\n\n // Close the Toast after the specified duration.\n this.doc_.getWin().setTimeout(() => {\n this.close();\n }, (toastDurationSeconds + 1) * 1000);\n }\n\n private async animate_({\n callback,\n }: {\n callback: () => Promise;\n }): Promise {\n // Wait for previous animations to finish.\n await this.animating_;\n\n try {\n await callback();\n } catch (err) {\n // Ignore errors to make sure animations don't get stuck.\n }\n }\n\n /**\n * Closes the toast.\n */\n close(): Promise {\n this.animating_ = this.animate_({\n callback: () => {\n // Remove the toast from the DOM after animation is complete.\n this.doc_.getWin().setTimeout(() => {\n this.doc_.getBody()?.removeChild(this.iframe_);\n }, 500);\n\n return transition(\n this.iframe_,\n {\n 'transform': 'translateY(100%)',\n 'opacity': '1',\n 'visibility': 'visible',\n },\n 400,\n 'ease-out'\n );\n },\n });\n\n return this.animating_;\n }\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Whether the specified error is an AbortError type.\n * See https://heycam.github.io/webidl/#aborterror.\n */\nexport function isCancelError(error: Error): boolean {\n return error?.name === 'AbortError';\n}\n\n/**\n * Creates an Error of AbortError type.\n * See https://heycam.github.io/webidl/#aborterror.\n */\nexport function createCancelError(message: string): Error {\n const error = new Error('AbortError: ' + message);\n error.name = 'AbortError';\n return error;\n}\n\n/**\n * A set of error utilities combined in a class to allow easy stubbing in tests.\n */\nexport class ErrorUtils {\n static throwAsync(error: Error) {\n setTimeout(() => {\n throw error;\n });\n }\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Debug logger, only log message if #swg.log=1 exists in URL.\n */\nexport function debugLog(...args: unknown[]) {\n if (/swg.debug=1/.test(self.location.hash)) {\n args.unshift('[Subscriptions]');\n log(...args);\n }\n}\n\nexport function log(...args: unknown[]) {\n // eslint-disable-next-line no-console\n console /*OK*/\n .log(...args);\n}\n\nexport function warn(...args: unknown[]) {\n // eslint-disable-next-line no-console\n console /*OK*/\n .warn(...args);\n}\n\n/**\n * Throws an error if the first argument isn't trueish.\n *\n * Supports argument substitution into the message via %s placeholders.\n * @param shouldBeTrueish The value to assert. The assert fails if it does\n * not evaluate to true.\n * @param message The assertion message\n * @param args Arguments substituted into %s in the message.\n */\nexport function assert(\n shouldBeTrueish: unknown,\n message = 'Assertion failed',\n ...args: unknown[]\n): void {\n if (shouldBeTrueish) {\n return;\n }\n\n const splitMessage = message.split('%s');\n const first = splitMessage.shift();\n let formatted = first;\n for (const arg of args) {\n const nextConstant = splitMessage.shift();\n formatted += toString(arg) + nextConstant;\n }\n throw new Error(formatted);\n}\n\nfunction toString(val: unknown): string {\n // Do check equivalent to `val instanceof Element` without cross-window bug\n const possibleElement = val as Element;\n if (possibleElement?.nodeType == 1) {\n return (\n possibleElement.tagName.toLowerCase() +\n (possibleElement.id ? '#' + possibleElement.id : '')\n );\n }\n return String(val);\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {Doc} from '../model/doc';\nimport {Message} from '../proto/api_messages';\nimport {warn} from './log';\n\n// NOTE: This regex was copied from SwG's AMP extension. https://github.com/ampproject/amphtml/blob/c23bf281f817a2ee5df73f6fd45e9f4b71bb68b6/extensions/amp-subscriptions-google/0.1/amp-subscriptions-google.js#L56\nconst GOOGLE_DOMAIN_RE = /(^|\\.)google\\.(com?|[a-z]{2}|com?\\.[a-z]{2}|cat)$/;\n\ninterface Location {\n href: string;\n protocol: string;\n host: string;\n hostname: string;\n port: string;\n pathname: string;\n search: string;\n hash: string;\n origin: string;\n}\n\n/**\n * Cached a-tag to avoid memory allocation during URL parsing.\n */\nconst a = self.document.createElement('a');\n\n/**\n * We cached all parsed URLs. As of now there are no use cases\n * of AMP docs that would ever parse an actual large number of URLs,\n * but we often parse the same one over and over again.\n */\nconst cache: {[url: string]: Location} = {};\n\n/**\n * Returns a Location-like object for the given URL. If it is relative,\n * the URL gets resolved.\n * Consider the returned object immutable. This is enforced during\n * testing by freezing the object.\n */\nexport function parseUrl(url: string): Location {\n const fromCache = cache[url];\n if (fromCache) {\n return fromCache;\n }\n\n const info = parseUrlWithA(a, url);\n\n return (cache[url] = info);\n}\n\n/**\n * Returns a Location-like object for the given URL. If it is relative,\n * the URL gets resolved.\n */\nfunction parseUrlWithA(a: HTMLAnchorElement, url: string): Location {\n a.href = url;\n\n const info: Location = {\n href: a.href,\n protocol: a.protocol,\n host: a.host,\n hostname: a.hostname,\n port: a.port == '0' ? '' : a.port,\n pathname: a.pathname,\n search: a.search,\n hash: a.hash,\n origin: a.protocol + '//' + a.host,\n };\n\n // For data URI a.origin is equal to the string 'null' which is not useful.\n // We instead return the actual origin which is the full URL.\n if (a.origin && a.origin !== 'null') {\n info.origin = a.origin;\n } else if (info.protocol === 'data:' || !info.host) {\n info.origin = info.href;\n }\n return info;\n}\n\n/**\n * Parses and builds Object of URL query string.\n * @param query The URL query string.\n */\nexport function parseQueryString(query: string): {[key: string]: string} {\n if (!query) {\n return {};\n }\n return (/^[?#]/.test(query) ? query.slice(1) : query)\n .split('&')\n .reduce((params, param) => {\n const item = param.split('=');\n try {\n const key = decodeURIComponent(item[0] || '');\n const value = decodeURIComponent(item[1] || '');\n if (key) {\n params[key] = value;\n }\n } catch (err) {\n // eslint-disable-next-line no-console\n warn(`SwG could not parse a URL query param: ${item[0]}`);\n }\n return params;\n }, {} as {[key: string]: string});\n}\n\n/**\n * Adds a parameter to a query string.\n */\nexport function addQueryParam(\n url: string,\n param: string,\n value: string\n): string {\n const queryIndex = url.indexOf('?');\n const fragmentIndex = url.indexOf('#');\n let fragment = '';\n if (fragmentIndex != -1) {\n fragment = url.substring(fragmentIndex);\n url = url.substring(0, fragmentIndex);\n }\n if (queryIndex == -1) {\n url += '?';\n } else if (queryIndex < url.length - 1) {\n url += '&';\n }\n url += encodeURIComponent(param) + '=' + encodeURIComponent(value);\n\n return url + fragment;\n}\n\nexport function serializeProtoMessageForUrl(message: Message): string {\n return JSON.stringify(message.toArray(false));\n}\n\nexport function getCanonicalTag(doc: Doc): string | undefined {\n const rootNode = doc.getRootNode();\n const canonicalTag = rootNode.querySelector(\n \"link[rel='canonical']\"\n ) as HTMLLinkElement;\n return canonicalTag?.href;\n}\n\n/**\n * Returns the canonical URL from the canonical tag. If the canonical tag is\n * not present, treat the doc URL itself as canonical.\n */\nexport function getCanonicalUrl(doc: Doc): string {\n const rootNode = doc.getRootNode();\n return (\n getCanonicalTag(doc) ||\n rootNode.location.origin + rootNode.location.pathname\n );\n}\n\nconst PARSED_URL = parseUrl(self.window.location.href);\nconst PARSED_REFERRER = parseUrl(self.document.referrer);\n\n/**\n * True for Google domains\n * @param parsedUrl Defaults to the current page's URL\n */\nfunction isGoogleDomain(parsedUrl: Location): boolean {\n return GOOGLE_DOMAIN_RE.test(parsedUrl.hostname);\n}\n\n/**\n * True for HTTPS URLs\n * @param parsedUrl Defaults to the current page's URL\n */\nexport function isSecure(parsedUrl = PARSED_URL): boolean {\n return parsedUrl.protocol === 'https' || parsedUrl.protocol === 'https:';\n}\n\n/**\n * True when the page is rendered within a secure Google application or\n * was linked to from a secure Google domain.\n * @param parsedReferrer Defaults to the current page's referrer\n */\nexport function wasReferredByGoogle(parsedReferrer = PARSED_REFERRER): boolean {\n return isSecure(parsedReferrer) && isGoogleDomain(parsedReferrer);\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {ErrorUtils} from '../utils/errors';\nimport {Message} from '../proto/api_messages';\nimport {parseUrl, serializeProtoMessageForUrl} from '../utils/url';\n\nconst jsonSaftyPrefix = /^(\\)\\]\\}'\\n)/;\n\nexport interface Fetcher {\n fetchCredentialedJson(url: string): Promise<{}>;\n\n fetch(url: string, init: RequestInit): Promise;\n\n /**\n * POST data to a URL endpoint, do not wait for a response.\n */\n sendBeacon(url: string, message: Message): void;\n\n /**\n * POST data to a URL endpoint, get a Promise for a response\n */\n sendPost(url: string, message: Message): Promise<{[key: string]: unknown}>;\n}\n\nexport class XhrFetcher implements Fetcher {\n constructor(private readonly win_: Window) {}\n\n async fetchCredentialedJson(url: string): Promise<{}> {\n const init: RequestInit = {\n method: 'GET',\n headers: {'Accept': 'text/plain, application/json'},\n credentials: 'include',\n };\n const response = await this.fetch(url, init);\n const text = await response.text();\n // Remove \"\")]}'\\n\" XSSI prevention prefix in safe responses.\n const cleanedText = text.replace(jsonSaftyPrefix, '');\n return JSON.parse(cleanedText);\n }\n\n async sendPost(\n url: string,\n message: Message\n ): Promise<{[key: string]: unknown}> {\n const init: RequestInit = {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8',\n },\n credentials: 'include',\n body: 'f.req=' + serializeProtoMessageForUrl(message),\n };\n const response = await this.fetch(url, init);\n if (!response) {\n return {};\n }\n\n const text = await response.text();\n try {\n // Remove \"\")]}'\\n\" XSSI prevention prefix in safe responses.\n const cleanedText = text.replace(jsonSaftyPrefix, '');\n return JSON.parse(cleanedText);\n } catch (err) {\n ErrorUtils.throwAsync(err as Error);\n return {};\n }\n }\n\n async fetch(url: string, init: RequestInit): Promise {\n try {\n // Wait for the request to succeed before returning the response,\n // allowing this method to catch failures.\n const response = await this.win_.fetch(url, init);\n return response;\n } catch (reason) {\n /*\n * If the domain is not valid for SwG we return 404 without\n * CORS headers and the browser throws a CORS error.\n * We include some helpful text in the message to point the\n * publisher towards the real problem.\n */\n const targetOrigin = parseUrl(url).origin;\n throw new Error(\n `XHR Failed fetching (${targetOrigin}/...): (Note: a CORS error above may indicate that this publisher or domain is not configured in Publisher Center. The CORS error happens because 4xx responses do not set CORS headers.)\\n\\n` +\n reason\n );\n }\n }\n\n sendBeacon(url: string, message: Message): void {\n const headers = {type: 'application/x-www-form-urlencoded;charset=UTF-8'};\n const blob = new Blob(\n ['f.req=' + serializeProtoMessageForUrl(message)],\n headers\n );\n navigator.sendBeacon(url, blob);\n }\n}\n","/**\n * Copyright 2018 The Subscribe with Google Authors. All Rights Reserved.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS-IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport {\n ADS_SERVER,\n FRONTEND,\n FRONTEND_CACHE,\n INTERNAL_RUNTIME_VERSION,\n PAY_ENVIRONMENT,\n PLAY_ENVIRONMENT,\n} from '../constants';\nimport {addQueryParam, parseQueryString, parseUrl} from '../utils/url';\n\n/** Possible cache keys which can influence how Swgjs busts caches for iframes. */\ntype CacheKey = 'zero' | 'nocache' | 'hr1' | 'hr12';\n\n/**\n * Have to put these in the map to avoid compiler optimization. Due to\n * optimization issues, this map only allows property-style keys. E.g. \"hr1\",\n * as opposed to \"1hr\".\n */\nexport const CACHE_KEYS: {[key in CacheKey]: string} = {\n 'zero': '0', //testing value\n 'nocache': '1',\n 'hr1': '3600000', // 1hr = 1000 * 60 * 60\n 'hr12': '43200000', // 12hr = 1000 * 60 * 60 * 12\n};\n\ninterface OperatingMode {\n frontEnd: string;\n payEnv: string;\n playEnv: string;\n feCache: CacheKey;\n}\n\n/**\n * Default operating Mode\n */\nexport const DEFAULT: OperatingMode = {\n frontEnd: FRONTEND,\n payEnv: PAY_ENVIRONMENT,\n playEnv: PLAY_ENVIRONMENT,\n feCache: FRONTEND_CACHE,\n};\n\n/**\n * Default operating Mode\n */\nconst PROD: OperatingMode = {\n frontEnd: 'https://news.google.com',\n payEnv: 'PRODUCTION',\n playEnv: 'PROD',\n feCache: 'nocache',\n};\n\n/**\n * Default operating Mode\n */\nconst AUTOPUSH: OperatingMode = {\n frontEnd: 'https://subscribe-autopush.sandbox.google.com',\n payEnv: 'PRODUCTION',\n playEnv: 'AUTOPUSH',\n feCache: 'nocache',\n};\n\n/**\n * Default operating Mode\n */\nconst QUAL: OperatingMode = {\n frontEnd: 'https://subscribe-qual.sandbox.google.com',\n payEnv: 'SANDBOX',\n playEnv: 'STAGING',\n feCache: 'nocache',\n};\n\n/**\n * Operating modes, only runtime switchable modes are here.\n * Build time modes set the default and are configured in prepare.sh.\n *\n * IMPORTANT: modes other than prod will only work on Google internal networks!\n */\nexport const MODES: {[key: string]: OperatingMode} = {\n 'default': DEFAULT,\n 'prod': PROD,\n 'autopush': AUTOPUSH,\n 'qual': QUAL,\n};\n\n/**\n * Check for swg.mode= in url fragment. If it exists, use it,\n * otherwise use the default build mode.\n */\nexport function getSwgMode(): OperatingMode {\n const query = parseQueryString(self.location.hash);\n const swgMode = query['swg.mode'];\n if (swgMode && MODES[swgMode]) {\n return MODES[swgMode];\n }\n return MODES['default'];\n}\n\nexport function feOrigin(): string {\n return parseUrl(getSwgMode().frontEnd).origin;\n}\n\n/**\n * @param url Relative URL, e.g. \"/service1\".\n * @return The complete URL.\n */\nexport function serviceUrl(url: string): string {\n // Allows us to make API calls with enabled experiments.\n const query = parseQueryString(self.location.hash);\n const experiments = query['swg.experiments'];\n if (experiments !== undefined) {\n url = addQueryParam(url, 'e', experiments);\n }\n\n return `${getSwgMode().frontEnd}/swg/_/api/v1` + url;\n}\n\n/**\n * @param url Relative URL, e.g. \"/service1\".\n * @return The complete URL.\n */\nexport function adsUrl(url: string): string {\n return ADS_SERVER + url;\n}\n\n/**\n * @param url Relative URL, e.g. \"/offersiframe\".\n * @param params List of extra params to append to the URL.\n * @param prefix\n * @return The complete URL.\n */\nexport function feUrl(\n url: string,\n params: {[key: string]: string} = {},\n prefix = ''\n): string {\n // Add cache param.\n const prefixed = prefix ? `swg/${prefix}` : 'swg';\n url = feCached(`${getSwgMode().frontEnd}/${prefixed}/ui/v1${url}`);\n\n // Optionally add jsmode param. This allows us to test against \"aggressively\" compiled Boq JS.\n const query = parseQueryString(self.location.hash);\n const boqJsMode = query['swg.boqjsmode'];\n if (boqJsMode !== undefined) {\n url = addQueryParam(url, 'jsmode', boqJsMode);\n }\n\n // Allows us to open iframes with enabled experiments.\n const experiments = query['swg.experiments'];\n if (experiments !== undefined) {\n url = addQueryParam(url, 'e', experiments);\n }\n\n for (const param in params) {\n url = addQueryParam(url, param, params[param]);\n }\n\n return url;\n}\n\n/**\n * @param url FE URL.\n * @return The complete URL including cache param.\n */\nexport function feCached(url: string): string {\n return addQueryParam(url, '_', cacheParam(getSwgMode().feCache));\n}\n\nexport function feArgs(args: {}): {} {\n return Object.assign(args, {\n '_client': `SwG ${INTERNAL_RUNTIME_VERSION}`,\n });\n}\n\nexport function cacheParam(cacheKey: CacheKey): string {\n const period = Number(CACHE_KEYS[cacheKey] || 1);\n if (period === 0) {\n return '_';\n }\n const now = Date.now();\n return String(period <= 1 ? now : Math.floor(now / period));\n}\n","/**\n * @license\n * SPDX-License-Identifier: Apache-2.0\n */\nimport '../environment/dev';\n/**\n * A secret token that must be passed to safe type constructors. It is only\n * accessible from within safevalues, ensuring that unrestricted safe type\n * creation is only possible within safevalues. In particular, this prevents\n * forgery such as `safeHtmlValue.constructor('javascript:evil')`.\n */\nexport const secretToken = {};\n/**\n * Asserts that the given token matches the secret safevalues token. An\n * exception is thrown if that is not the case.\n */\nexport function ensureTokenIsValid(token) {\n if (process.env.NODE_ENV !== 'production') {\n if (token !== secretToken) {\n throw new Error('Bad secret');\n }\n }\n}\n","/**\n * @license\n * SPDX-License-Identifier: Apache-2.0\n */\n/**\n * The name of the Trusted Types policy used by TS safevalues, or empty\n * to disable Trusted Types. This duplicates the 'google#safe', but\n * can be overridden in tests.\n */\nlet trustedTypesPolicyName = 'google#safe';\n/** Helper to retrieve the value of `window.trustedTypes`. */\nfunction trustedTypes() {\n if (typeof window !== 'undefined') {\n return window.trustedTypes;\n }\n return undefined;\n}\n/**\n * Returns window.trustedTypes if Trusted Types are enabled and supported, or\n * null otherwise.\n */\nexport function getTrustedTypes() {\n var _a;\n return (trustedTypesPolicyName !== '') ? ((_a = trustedTypes()) !== null && _a !== void 0 ? _a : null) : null;\n}\n/**\n * The Trusted Types policy used by TS safevalues, or null if Trusted Types\n * are not enabled/supported, or undefined if the policy has not been created\n * yet.\n */\nlet trustedTypesPolicy;\n/**\n * Returns the Trusted Types policy used by TS safevalues, or null if Trusted\n * Types are not enabled/supported. The first call to this function will\n * create the policy.\n */\nexport function getTrustedTypesPolicy() {\n var _a, _b;\n if (trustedTypesPolicy === undefined) {\n try {\n trustedTypesPolicy =\n (_b = (_a = getTrustedTypes()) === null || _a === void 0 ? void 0 : _a.createPolicy(trustedTypesPolicyName, {\n createHTML: (s) => s,\n createScript: (s) => s,\n createScriptURL: (s) => s\n })) !== null && _b !== void 0 ? _b : null;\n }\n catch (_c) {\n // In Chromium versions before 81, trustedTypes.createPolicy throws if\n // called with a name that is already registered, even if no CSP is set.\n // Until users have largely migrated to 81 or above, catch the error not\n // to break the applications functionally. In such case, the code will\n // fall back to using regular Safe Types.\n trustedTypesPolicy = null;\n }\n }\n return trustedTypesPolicy;\n}\n/** Helpers for tests. */\nexport const TEST_ONLY = {\n resetDefaults() {\n trustedTypesPolicy = undefined;\n trustedTypesPolicyName = 'google#safe';\n },\n setTrustedTypesPolicyName(name) {\n trustedTypesPolicyName = name;\n },\n};\n","/**\n * @license\n * SPDX-License-Identifier: Apache-2.0\n */\nimport '../environment/dev';\nimport { ensureTokenIsValid, secretToken } from './secrets';\nimport { getTrustedTypes, getTrustedTypesPolicy } from './trusted_types';\n/**\n * Runtime implementation of `TrustedHTML` in browsers that don't support it.\n */\nclass HtmlImpl {\n constructor(html, token) {\n ensureTokenIsValid(token);\n this.privateDoNotAccessOrElseWrappedHtml = html;\n }\n toString() {\n return this.privateDoNotAccessOrElseWrappedHtml.toString();\n }\n}\nfunction createTrustedHtmlOrPolyfill(html, trusted) {\n return (trusted !== null && trusted !== void 0 ? trusted : new HtmlImpl(html, secretToken));\n}\nconst GlobalTrustedHTML = (typeof window !== 'undefined') ? window.TrustedHTML : undefined;\n/**\n * Also exports the constructor so that instanceof checks work.\n */\nexport const SafeHtml = (GlobalTrustedHTML !== null && GlobalTrustedHTML !== void 0 ? GlobalTrustedHTML : HtmlImpl);\n/**\n * Builds a new `SafeHtml` from the given string, without enforcing safety\n * guarantees. It may cause side effects by creating a Trusted Types policy.\n * This shouldn't be exposed to application developers, and must only be used as\n * a step towards safe builders or safe constants.\n */\nexport function createHtmlInternal(html) {\n var _a;\n /** @noinline */\n const noinlineHtml = html;\n return createTrustedHtmlOrPolyfill(noinlineHtml, (_a = getTrustedTypesPolicy()) === null || _a === void 0 ? void 0 : _a.createHTML(noinlineHtml));\n}\n/**\n * An empty `SafeHtml` constant.\n * Unlike the function above, using this will not create a policy.\n */\nexport const EMPTY_HTML = \n/* #__PURE__ */ (() => { var _a; return createTrustedHtmlOrPolyfill('', (_a = getTrustedTypes()) === null || _a === void 0 ? void 0 : _a.emptyHTML); })();\n/**\n * Checks if the given value is a `SafeHtml` instance.\n */\nexport function isHtml(value) {\n var _a;\n return ((_a = getTrustedTypes()) === null || _a === void 0 ? void 0 : _a.isHTML(value)) || value instanceof HtmlImpl;\n}\n/**\n * Returns the value of the passed `SafeHtml` object while ensuring it\n * has the correct type.\n *\n * Returns a native `TrustedHTML` or a string if Trusted Types are disabled.\n */\nexport function unwrapHtml(value) {\n var _a;\n if ((_a = getTrustedTypes()) === null || _a === void 0 ? void 0 : _a.isHTML(value)) {\n return value;\n }\n else if (value instanceof HtmlImpl) {\n return value.privateDoNotAccessOrElseWrappedHtml;\n }\n else {\n let message = '';\n if (process.env.NODE_ENV !== 'production') {\n message = 'Unexpected type when unwrapping SafeHtml';\n }\n throw new Error(message);\n }\n}\n","/**\n * @license\n * SPDX-License-Identifier: Apache-2.0\n */\nimport { createHtmlInternal, isHtml, unwrapHtml } from '../internals/html_impl';\nimport { unwrapResourceUrl } from '../internals/resource_url_impl';\nimport { unwrapScript } from '../internals/script_impl';\n/**\n * Returns HTML-escaped text as a `SafeHtml` object. No-op if value is already a\n * SafeHtml instance.\n *\n * Available options:\n * - `preserveSpaces` turns every second consecutive space character into its\n * HTML entity representation (` `).\n * - `preserveNewlines` turns newline characters into breaks (`
`).\n * - `preserveTabs` wraps tab characters in a span with style=white-space:pre.\n */\nexport function htmlEscape(value, options = {}) {\n if (isHtml(value)) {\n return value;\n }\n let htmlEscapedString = htmlEscapeToString(String(value));\n if (options.preserveSpaces) {\n // Do this first to ensure we preserve spaces after newlines and tabs.\n htmlEscapedString =\n htmlEscapedString.replace(/(^|[\\r\\n\\t ]) /g, '$1 ');\n }\n if (options.preserveNewlines) {\n htmlEscapedString = htmlEscapedString.replace(/(\\r\\n|\\n|\\r)/g, '
');\n }\n if (options.preserveTabs) {\n htmlEscapedString = htmlEscapedString.replace(/(\\t+)/g, '$1');\n }\n return createHtmlInternal(htmlEscapedString);\n}\n/**\n * Creates a `SafeHtml` representing a script tag with inline script content.\n */\nexport function scriptToHtml(script, options = {}) {\n const unwrappedScript = unwrapScript(script).toString();\n let stringTag = `