{ "version": 3, "sources": ["../../account/push_notifications.ts", "../../cling_button_gateway.ts"], "sourcesContent": ["import {report_error, report_info} from \"@cling/lib.shared.debug\"\nimport {i18n} from \"@cling/lib.web.i18n\"\nimport {promp_state} from \"../dialogs/prompt\"\nimport {current_user, ui_actions} from \"../state\"\nimport {call_function} from \"@cling/lib.shared.faas\"\nimport {\n PatchAccountSettings,\n PatchFullAccountRequest,\n PushSubscription as ModelPushSubscription,\n create_PatchUID,\n} from \"@cling/lib.shared.model\"\nimport {log} from \"@cling/lib.shared.logging\"\nimport {register_logout_listener} from \"@cling/lib.web.auth\"\nimport {goto_board} from \"../utils\"\nimport {safe_local_storage} from \"@cling/lib.web.utils/safe_storage\"\nimport {not_null} from \"@cling/lib.shared.utils\"\nimport {PushNavigateMessage} from \"@cling/lib.web.service_worker/message_types\"\n\nconst vapid_public_key = not_null(\n location.hostname.startsWith(\"dev-pero\")\n ? \"BEw7970HhvA0oBtbNePJ8D5stumaUATeUfe2MKhZzdWXOLB_mhAIaAz70Ky4-VY0gA7dWdwUI4BO1TrH1VuVMW0\"\n : location.hostname.startsWith(\"dev-jenkins\")\n ? \"BFMGB39Hh8mB-Fk1t2ZpXFYvteXlHZcAnSCrtJPZTACF0tlGifm0k5qgUKgBHfP7808qSiB7TCrNaVaki7wtEVA\"\n : location.hostname.startsWith(\"cling.com\")\n ? \"BHwCBNvDTdPdV3LGguO9uv8tgXgvd1pi6vu7A-w0zYJcvTOPoLRopaZmcnQKg5LRv4BM4IzRKx5nO8e4-C37OC8\"\n : process.env.NODE_ENV === \"test\"\n ? \"test\"\n : undefined,\n \"No VAPID public key for the current hostname\",\n)\n\nlet _init_called = false\nexport let push_available = false\nexport function init() {\n if (_init_called) {\n return\n }\n _init_called = true\n push_available = \"serviceWorker\" in navigator && \"PushManager\" in window\n if (!push_available) {\n log.info(\"Push notifications are not available\")\n return\n }\n register_logout_listener(unsubscribe)\n navigator.serviceWorker.addEventListener(\"message\", (event: {data: PushNavigateMessage}) => {\n if (event.data.type !== \"push_navigate\") {\n return\n }\n goto_board({\n board_uid: event.data.board_uid,\n highlight_card_uid: event.data.card_uid,\n })\n .then(() => {\n if (event.data.open_comments && event.data.card_uid) {\n ui_actions.open_comments(event.data.card_uid)\n }\n })\n .catch(report_error)\n })\n}\n\n/**\n * Ask the user for permission to send push notifications and then subscribe and send\n * the subscription to the server.\n */\nexport async function subscribe_flow() {\n if (Notification.permission !== \"granted\") {\n await new Promise<void>((resolve, reject) => {\n promp_state.show_prompt({\n title: i18n.permission_required,\n content: i18n.your_browser_will_ask_for_permission_to_send_notifications,\n desktop_focus: \"accept\",\n button_label_accept: i18n.ok,\n button_label_cancel: i18n.cancel,\n onAccept: async () => {\n try {\n await Notification.requestPermission()\n resolve()\n } catch (error) {\n report_error(\n \"Failed to request push notification permission - showing snackbar\",\n error,\n )\n reject(error)\n }\n },\n })\n })\n }\n const subscription = await subscribe()\n if (\n current_user.account_settings.board_changed_push_subscriptions.find(\n (x) => x.endpoint === subscription.endpoint,\n )\n ) {\n return\n }\n const subscription_json = subscription.toJSON()\n const push_subscription = new ModelPushSubscription({\n endpoint: subscription.endpoint,\n auth: subscription_json.keys!.auth,\n p256dh: subscription_json.keys!.p256dh,\n })\n await call_function(\n new PatchFullAccountRequest({\n patch_uid: create_PatchUID(),\n patch_account_settings: new PatchAccountSettings({\n add_board_changed_push_subscription: push_subscription,\n }),\n }),\n )\n}\n\nasync function subscribe(): Promise<PushSubscription> {\n const service_worker_registration = await navigator.serviceWorker.ready\n const subscription = await service_worker_registration.pushManager.subscribe({\n userVisibleOnly: true,\n applicationServerKey: vapid_public_key,\n })\n safe_local_storage.setItem(\"last_push_subscription_refresh\", Date.now().toString())\n return subscription\n}\n\n/**\n * Unsubscribe the user from push notifications locally and on the server if possible.\n *\n * This function can be called even if there is no service worker registration.\n */\nexport async function unsubscribe() {\n if (!push_available) {\n return\n }\n const registrations = await navigator.serviceWorker.getRegistrations()\n safe_local_storage.removeItem(\"pending_refresh_push_subscription\")\n if (registrations.length === 0) {\n log.debug(\n \"No service worker registrations found - cannot unsubscribe from push notifications\",\n )\n return\n }\n const service_worker_registration = await navigator.serviceWorker.ready\n let subscription\n try {\n subscription = await service_worker_registration.pushManager.getSubscription()\n } catch (error) {\n // In incognito mode, we get an error when trying to get the subscription.\n log.debug(\"Failed to get push subscription - nothing more to do\", error)\n return\n }\n await subscription?.unsubscribe()\n if (subscription && current_user) {\n // If we are logged in, we also remove the subscription from the server.\n const subscription_json = subscription.toJSON()\n if (\n current_user.account_settings.board_changed_push_subscriptions.find(\n (x) => x.endpoint === subscription.endpoint,\n )\n ) {\n await call_function(\n new PatchFullAccountRequest({\n patch_uid: create_PatchUID(),\n patch_account_settings: new PatchAccountSettings({\n remove_board_changed_push_subscription: new ModelPushSubscription({\n endpoint: subscription_json.endpoint!,\n auth: subscription_json.keys!.auth,\n p256dh: subscription_json.keys!.p256dh,\n }),\n }),\n }),\n )\n }\n }\n}\n\n/**\n * Check if the user is subscribed by first seeing if there is a local `pushManager.getSubscription()`.\n * If there is one, we check if the subscription is already stored in the user's account settings.\n * If not, we unsubscribe the user.\n */\nexport async function is_subscribed(): Promise<boolean> {\n const service_worker_registration = await navigator.serviceWorker.ready\n const subscription = await service_worker_registration.pushManager.getSubscription()\n if (!subscription) {\n return false\n }\n if (\n current_user.account_settings.board_changed_push_subscriptions.find(\n (x) => x.endpoint === subscription.endpoint,\n )\n ) {\n return true\n }\n report_info(\n \"Unsubscribing user because the subscription is not stored in the user's account settings\",\n )\n await subscription.unsubscribe()\n return false\n}\n", "import type {MessageFromWebApp, TokensMessage} from \"@cling/lib.shared.types/message_types\"\nimport {register_auth_listener} from \"@cling/lib.web.auth\"\nimport {report_error} from \"@cling/lib.shared.debug\"\n\nlet _tokens_message: TokensMessage\nlet _cling_button_hub_mutation_observer: MutationObserver\n\nexport function init(silent_login: boolean) {\n register_auth_listener(({account_uid, access_token, access_token_expires_at}) => {\n // Note: We use a variable at module scope here, because the `MutationObserver`\n // created below, should send the latest tokens, if it detects that the\n // Cling Button is listening for messages ...\n _tokens_message = {\n id: \"tokens\",\n account_uid,\n id_token: access_token,\n access_token,\n access_token_expires_at,\n silent_login,\n }\n const cling_button_hub = document.getElementById(\"cling-button-hub\") as HTMLElement\n if (cling_button_hub.classList.contains(\"cling-button-is-listening\")) {\n // The Cling Button is already listening ...\n send_message_to_browser_extension(_tokens_message)\n } else if (!_cling_button_hub_mutation_observer) {\n // The Cling Button is not listening yet\n // *and* no MutationObserver is installed yet ...\n _cling_button_hub_mutation_observer = new MutationObserver(() => {\n if (cling_button_hub.classList.contains(\"cling-button-is-listening\")) {\n send_message_to_browser_extension(_tokens_message)\n _cling_button_hub_mutation_observer.disconnect()\n }\n })\n _cling_button_hub_mutation_observer.observe(cling_button_hub, {\n attributes: true,\n })\n }\n })\n}\n\nexport function logout() {\n if (_tokens_message) {\n send_message_to_browser_extension({\n id: \"logout\",\n })\n }\n}\n\nfunction send_message_to_browser_extension(message: MessageFromWebApp) {\n // Note: We don't want to send any messages to the Cling Button\n // from a Cling web app running on a development server ...\n if (cling.dev) {\n return\n }\n try {\n postMessage(message, `https://${location.host}/`)\n } catch (error) {\n report_error(error)\n }\n}\n"], "mappings": "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", "names": ["vapid_public_key", "not_null", "_init_called", "push_available", "init", "log", "register_logout_listener", "unsubscribe", "event", "goto_board", "ui_actions", "report_error", "__name", "subscribe_flow", "resolve", "reject", "promp_state", "i18n", "error", "subscription", "subscribe", "current_user", "x", "subscription_json", "push_subscription", "PushSubscription", "call_function", "PatchFullAccountRequest", "create_PatchUID", "PatchAccountSettings", "safe_local_storage", "registrations", "service_worker_registration", "is_subscribed", "report_info", "_tokens_message", "_cling_button_hub_mutation_observer", "init", "silent_login", "register_auth_listener", "account_uid", "access_token", "access_token_expires_at", "cling_button_hub", "send_message_to_browser_extension", "__name", "logout", "message", "error", "report_error"] }