summaryrefslogtreecommitdiffstats
path: root/toolkit/components/aboutwebauthn/content
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--toolkit/components/aboutwebauthn/content/aboutWebauthn.css99
-rw-r--r--toolkit/components/aboutwebauthn/content/aboutWebauthn.html322
-rw-r--r--toolkit/components/aboutwebauthn/content/aboutWebauthn.js866
3 files changed, 1287 insertions, 0 deletions
diff --git a/toolkit/components/aboutwebauthn/content/aboutWebauthn.css b/toolkit/components/aboutwebauthn/content/aboutWebauthn.css
new file mode 100644
index 0000000000..5654e4276b
--- /dev/null
+++ b/toolkit/components/aboutwebauthn/content/aboutWebauthn.css
@@ -0,0 +1,99 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+@import url("chrome://global/skin/in-content/common.css");
+
+html {
+ height: 100%;
+}
+
+body {
+ display: flex;
+ align-items: stretch;
+ height: 100%;
+}
+
+label {
+ display: block;
+}
+
+#info-text-div {
+ padding: 20px;
+}
+
+#ctap-listen-div {
+ padding-top: 15px;
+}
+#ctap-listen-result {
+ font-weight: 600;
+ font-size: 1.5em;
+ padding-inline-start: 15px;
+ height: 1.5em;
+
+ &.success {
+ background-color: var(--green-50);
+ }
+
+ &.error {
+ background-color: var(--red-50);
+ }
+}
+
+.category {
+ cursor: pointer;
+ /* Center category names */
+ display: flex;
+ align-items: center;
+}
+
+.optional-category {
+ display: none;
+}
+
+.disabled-category {
+ pointer-events: none;
+}
+
+@media (max-width: 830px){
+ #categories > .category {
+ padding-inline-start: 5px;
+ margin-inline-start: 0;
+ }
+}
+
+#main-content {
+ flex: 1;
+}
+
+.token-info-flex-box {
+ display: flex;
+}
+
+.token-info-flex-child {
+ flex: 1;
+}
+
+.token-info-flex-child#authenticator-options {
+ margin-inline-end: 2px;
+}
+
+.bio-enrollment-sample {
+ display: flex;
+ gap: 0.5em;
+}
+
+.button-row {
+ display: inline-block;
+ margin-inline-end: 5px;
+}
+
+.delete-icon {
+ margin-inline-end: 5px;
+ -moz-context-properties: fill;
+ fill: currentColor;
+}
+
+.delete-button {
+ display: inline-flex;
+}
diff --git a/toolkit/components/aboutwebauthn/content/aboutWebauthn.html b/toolkit/components/aboutwebauthn/content/aboutWebauthn.html
new file mode 100644
index 0000000000..5c8485b61f
--- /dev/null
+++ b/toolkit/components/aboutwebauthn/content/aboutWebauthn.html
@@ -0,0 +1,322 @@
+<!DOCTYPE html>
+
+<!-- This Source Code Form is subject to the terms of the Mozilla Public
+ - License, v. 2.0. If a copy of the MPL was not distributed with this
+ - file, You can obtain one at http://mozilla.org/MPL/2.0/. -->
+
+<html>
+ <head>
+ <meta
+ http-equiv="Content-Security-Policy"
+ content="default-src chrome:; style-src chrome:; object-src 'none'"
+ />
+ <meta charset="utf-8" />
+ <title id="page-title" data-l10n-id="about-webauthn-page-title"></title>
+ <link rel="stylesheet" href="chrome://global/content/aboutWebauthn.css" />
+ <script src="chrome://global/content/aboutWebauthn.js"></script>
+ <link rel="localization" href="toolkit/about/aboutWebauthn.ftl" />
+ </head>
+
+ <body id="body">
+ <div id="categories" role="tablist" aria-labelledby="page-title">
+ <div
+ id="info-tab-button"
+ class="category"
+ selected="true"
+ role="tab"
+ tabindex="0"
+ aria-selected="true"
+ aria-controls="token-info-section"
+ >
+ <span
+ class="tablinks"
+ data-l10n-id="about-webauthn-info-section-title"
+ ></span>
+ </div>
+ <div
+ id="pin-tab-button"
+ class="category optional-category"
+ role="tab"
+ tabindex="-1"
+ aria-selected="false"
+ aria-controls="set-change-pin-section"
+ >
+ <span
+ class="tablinks"
+ data-l10n-id="about-webauthn-pin-section-title"
+ ></span>
+ </div>
+ <div
+ id="credentials-tab-button"
+ class="category optional-category"
+ role="tab"
+ tabindex="-1"
+ aria-selected="false"
+ aria-controls="credential-management-section"
+ >
+ <span
+ class="tablinks"
+ data-l10n-id="about-webauthn-credential-management-section-title"
+ ></span>
+ </div>
+ <div
+ id="bio-enrollments-tab-button"
+ class="category optional-category"
+ role="tab"
+ tabindex="-1"
+ aria-selected="false"
+ aria-controls="bio-enrollment-section"
+ >
+ <span
+ class="tablinks"
+ data-l10n-id="about-webauthn-bio-enrollment-section-title"
+ ></span>
+ </div>
+ </div>
+
+ <div id="main-content">
+ <div id="ctap-listen-div">
+ <label id="ctap-listen-result"></label>
+ </div>
+
+ <div
+ class="tabcontent token-info-section"
+ id="token-info-section"
+ role="tabpanel"
+ aria-labelledby="info-tab-button"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-info-section-title"
+ ></h2>
+
+ <div id="info-text-div">
+ <label
+ id="info-text-field"
+ data-l10n-id="about-webauthn-text-connect-device"
+ ></label>
+ </div>
+
+ <div id="ctap2-token-info" class="token-info-flex-box" display="none">
+ <div id="ctap2-token-info-options" class="token-info-flex-child">
+ <h3 data-l10n-id="about-webauthn-options-subsection-title"></h3>
+ <table id="authenticator-options"></table>
+ </div>
+ <div id="ctap2-token-info-info" class="token-info-flex-child">
+ <h3 data-l10n-id="about-webauthn-info-subsection-title"></h3>
+ <table id="authenticator-info"></table>
+ </div>
+ </div>
+ </div>
+
+ <div
+ hidden
+ class="tabcontent pin-section"
+ id="set-change-pin-section"
+ role="tabpanel"
+ aria-labelledby="pin-tab-button"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-pin-section-title"
+ ></h2>
+ <div id="new-pin-div">
+ <label
+ for="new-pin"
+ data-l10n-id="about-webauthn-new-pin-label"
+ ></label>
+ <input type="password" id="new-pin" name="new-pin" required />
+ <label
+ for="new-pin-repeat"
+ data-l10n-id="about-webauthn-repeat-pin-label"
+ ></label>
+ <input
+ type="password"
+ id="new-pin-repeat"
+ name="new-pin-repeat"
+ required
+ />
+ </div>
+ <div id="current-pin-div">
+ <label
+ for="current-pin"
+ data-l10n-id="about-webauthn-current-pin-label"
+ ></label>
+ <input type="password" id="current-pin" name="current-pin" required />
+ </div>
+ <button
+ disabled
+ id="set-pin-button"
+ data-l10n-id="about-webauthn-current-set-pin-button"
+ ></button>
+ <button
+ disabled
+ id="change-pin-button"
+ data-l10n-id="about-webauthn-current-change-pin-button"
+ ></button>
+ <label id="set-change-pin-result" class="ctap-result"></label>
+ </div>
+
+ <div
+ hidden
+ class="tabcontent credential-management-section"
+ id="credential-management-section"
+ role="tabpanel"
+ aria-labelledby="credentials-tab-button"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-credential-management-section-title"
+ ></h2>
+ <div
+ hidden
+ id="credential-list-subsection"
+ class="token-info-flex-child"
+ >
+ <h3
+ data-l10n-id="about-webauthn-credential-list-subsection-title"
+ ></h3>
+ <div hidden id="credential-list-empty-label">
+ <label
+ hidden
+ data-l10n-id="about-webauthn-credential-list-empty"
+ ></label>
+ </div>
+ <table id="credential-list"></table>
+ </div>
+ <button
+ class="credentials-button"
+ id="list-credentials-button"
+ data-l10n-id="about-webauthn-list-credentials-button"
+ ></button>
+ </div>
+
+ <div
+ hidden
+ class="tabcontent bio-enrollment-section"
+ id="bio-enrollment-section"
+ role="tabpanel"
+ aria-labelledby="bio-enrollments-tab-button"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-bio-enrollment-section-title"
+ ></h2>
+ <div
+ hidden
+ id="bio-enrollment-list-subsection"
+ class="token-info-flex-child"
+ >
+ <h3
+ data-l10n-id="about-webauthn-bio-enrollment-list-subsection-title"
+ ></h3>
+ <div hidden id="bio-enrollment-list-empty-label">
+ <label
+ hidden
+ data-l10n-id="about-webauthn-enrollment-list-empty"
+ ></label>
+ </div>
+ <table id="bio-enrollment-list"></table>
+ </div>
+ <button
+ class="bio-enrollment-button button-row"
+ id="list-bio-enrollments-button"
+ data-l10n-id="about-webauthn-list-bio-enrollments-button"
+ ></button>
+ <button
+ class="bio-enrollment-button button-row"
+ id="add-bio-enrollment-button"
+ data-l10n-id="about-webauthn-add-bio-enrollment-button"
+ ></button>
+ </div>
+ <div
+ hidden
+ class="tabcontent add-bio-enrollment-section"
+ id="add-bio-enrollment-section"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-add-bio-enrollment-section-title"
+ ></h2>
+ <label
+ for="enrollment-name"
+ data-l10n-id="about-webauthn-enrollment-name-label"
+ ></label>
+ <input id="enrollment-name" name="enrollment-name" autofocus />
+ <button
+ class="bio-enrollment-button button-row"
+ id="start-enrollment-button"
+ data-l10n-id="about-webauthn-start-enrollment-button"
+ ></button>
+ <button
+ id="cancel-enrollment-button"
+ class="button-row"
+ data-l10n-id="about-webauthn-cancel-button"
+ ></button>
+ <div id="enrollment-update"></div>
+ </div>
+
+ <div
+ hidden
+ class="tabcontent pin-required-section"
+ id="pin-required-section"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-pin-required-section-title"
+ ></h2>
+ <div id="pin-div">
+ <label
+ for="pin-required"
+ data-l10n-id="about-webauthn-pin-required-label"
+ ></label>
+ <input
+ type="password"
+ id="pin-required"
+ name="pin-required"
+ required
+ autofocus
+ />
+ </div>
+ <button
+ id="send-pin-button"
+ class="button-row"
+ data-l10n-id="about-webauthn-send-pin-button"
+ ></button>
+ <button
+ id="cancel-send-pin-button"
+ class="button-row"
+ data-l10n-id="about-webauthn-cancel-button"
+ ></button>
+ </div>
+
+ <div
+ hidden
+ class="tabcontent confirm-deletion-section"
+ id="confirm-deletion-section"
+ >
+ <h2
+ class="categoryTitle"
+ data-l10n-id="about-webauthn-confirm-deletion-section-title"
+ ></h2>
+ <div id="confirmation-div">
+ <label
+ for="confirmation-context"
+ data-l10n-id="about-webauthn-confirm-deletion-label"
+ ></label>
+ <label id="confirmation-context"></label>
+ </div>
+ <button
+ id="confirm-deletion-button"
+ class="button-row"
+ data-l10n-id="about-webauthn-delete-button"
+ ></button>
+ <button
+ id="cancel-confirmation-button"
+ class="button-row"
+ data-l10n-id="about-webauthn-cancel-button"
+ ></button>
+ </div>
+ </div>
+ </body>
+</html>
diff --git a/toolkit/components/aboutwebauthn/content/aboutWebauthn.js b/toolkit/components/aboutwebauthn/content/aboutWebauthn.js
new file mode 100644
index 0000000000..9142ed198a
--- /dev/null
+++ b/toolkit/components/aboutwebauthn/content/aboutWebauthn.js
@@ -0,0 +1,866 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+"use strict";
+
+let AboutWebauthnService = null;
+
+var AboutWebauthnManagerJS = {
+ _topic: "about-webauthn-prompt",
+ _initialized: false,
+ _l10n: null,
+ _bio_l10n: null,
+ _curr_data: null,
+ _current_tab: "",
+ _previous_tab: "",
+
+ init() {
+ if (this._initialized) {
+ return;
+ }
+ this._l10n = new Localization(["toolkit/about/aboutWebauthn.ftl"], true);
+ this._bio_l10n = new Map();
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpGood",
+ "about-webauthn-ctap2-enroll-feedback-good"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooHigh",
+ "about-webauthn-ctap2-enroll-feedback-too-high"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooLow",
+ "about-webauthn-ctap2-enroll-feedback-too-low"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooLeft",
+ "about-webauthn-ctap2-enroll-feedback-too-left"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooRight",
+ "about-webauthn-ctap2-enroll-feedback-too-right"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooFast",
+ "about-webauthn-ctap2-enroll-feedback-too-fast"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooSlow",
+ "about-webauthn-ctap2-enroll-feedback-too-slow"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpPoorQuality",
+ "about-webauthn-ctap2-enroll-feedback-poor-quality"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooSkewed",
+ "about-webauthn-ctap2-enroll-feedback-too-skewed"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpTooShort",
+ "about-webauthn-ctap2-enroll-feedback-too-short"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpMergeFailure",
+ "about-webauthn-ctap2-enroll-feedback-merge-failure"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackFpExists",
+ "about-webauthn-ctap2-enroll-feedback-exists"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackNoUserActivity",
+ "about-webauthn-ctap2-enroll-feedback-no-user-activity"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackNoUserPresenceTransition",
+ "about-webauthn-ctap2-enroll-feedback-no-user-presence-transition"
+ );
+ this._bio_l10n.set(
+ "Ctap2EnrollFeedbackOther",
+ "about-webauthn-ctap2-enroll-feedback-other"
+ );
+
+ Services.obs.addObserver(this, this._topic);
+ this._initialized = true;
+ reset_page();
+ },
+
+ uninit() {
+ Services.obs.removeObserver(this, this._topic);
+ this._initialized = false;
+ this._l10n = null;
+ this._current_tab = "";
+ this._previous_tab = "";
+ },
+
+ observe(aSubject, aTopic, aData) {
+ let data = JSON.parse(aData);
+
+ // We have token
+ if (data.type == "selected-device") {
+ this._curr_data = data.auth_info;
+ this.show_ui_based_on_authenticator_info(data);
+ fake_click_event_for_id("info-tab-button");
+ } else if (data.type == "select-device") {
+ set_info_text("about-webauthn-text-select-device");
+ } else if (data.type == "pin-required") {
+ open_pin_required_tab();
+ } else if (data.type == "pin-invalid") {
+ let retries = data.retries ? data.retries : 0;
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-invalid-error",
+ JSON.stringify({ retriesLeft: retries })
+ );
+ open_pin_required_tab();
+ } else if (data.type == "bio-enrollment-update") {
+ if (data.result.EnrollmentList) {
+ show_results_banner("success", "about-webauthn-results-success");
+ this.show_enrollment_list(data.result.EnrollmentList);
+ bio_enrollment_in_progress(false);
+ } else if (data.result.DeleteSuccess || data.result.AddSuccess) {
+ show_results_banner("success", "about-webauthn-results-success");
+ clear_bio_enrollment_samples();
+ // Update AuthenticatorInfo
+ this._curr_data = data.result.DeleteSuccess ?? data.result.AddSuccess;
+ fake_click_event_for_id("bio-enrollments-tab-button");
+ bio_enrollment_in_progress(false);
+ // If we still have some enrollments to show, update the list
+ // otherwise, remove it.
+ if (
+ this._curr_data.options.bioEnroll === true ||
+ this._curr_data.options.userVerificationMgmtPreview === true
+ ) {
+ list_bio_enrollments();
+ } else {
+ // Hide the list, because it's empty
+ document.getElementById(
+ "bio-enrollment-list-subsection"
+ ).hidden = true;
+ }
+ } else if (data.result.UpdateSuccess) {
+ fake_click_event_for_id("bio-enrollments-tab-button");
+ list_bio_enrollments();
+ } else if (data.result.SampleStatus) {
+ show_add_bio_enrollment_section();
+ let up = document.getElementById("enrollment-update");
+ let sample_update = document.createElement("div");
+ let new_line = document.createElement("label");
+ new_line.classList.add("sample");
+ new_line.setAttribute(
+ "data-l10n-id",
+ this._bio_l10n.get(data.result.SampleStatus[0])
+ );
+ sample_update.appendChild(new_line);
+ let samples_needed_line = document.createElement("label");
+ samples_needed_line.setAttribute(
+ "data-l10n-id",
+ "about-webauthn-samples-still-needed"
+ );
+ samples_needed_line.setAttribute(
+ "data-l10n-args",
+ JSON.stringify({ repeatCount: data.result.SampleStatus[1] })
+ );
+ sample_update.classList.add("bio-enrollment-sample");
+ sample_update.appendChild(samples_needed_line);
+ up.appendChild(sample_update);
+ }
+ } else if (data.type == "credential-management-update") {
+ credential_management_in_progress(false);
+ if (data.result.CredentialList) {
+ show_results_banner("success", "about-webauthn-results-success");
+ this.show_credential_list(data.result.CredentialList.credential_list);
+ } else {
+ // DeleteSuccess or UpdateSuccess
+ show_results_banner("success", "about-webauthn-results-success");
+ list_credentials();
+ }
+ } else if (data.type == "listen-success") {
+ reset_page();
+ // Show results
+ show_results_banner("success", "about-webauthn-results-success");
+ this._reset_in_progress = "";
+ AboutWebauthnService.listen();
+ } else if (data.type == "listen-error") {
+ reset_page();
+
+ if (!data.error) {
+ show_results_banner("error", "about-webauthn-results-general-error");
+ } else if (data.error.type == "pin-auth-blocked") {
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-auth-blocked-error"
+ );
+ } else if (data.error.type == "pin-not-set") {
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-not-set-error"
+ );
+ } else if (data.error.type == "device-blocked") {
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-blocked-error"
+ );
+ } else if (data.error.type == "pin-is-too-short") {
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-too-short-error"
+ );
+ } else if (data.error.type == "pin-is-too-long") {
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-too-long-error"
+ );
+ } else if (data.error.type == "pin-invalid") {
+ let retries = data.error.retries
+ ? JSON.stringify({ retriesLeft: data.error.retries })
+ : null;
+ show_results_banner(
+ "error",
+ "about-webauthn-results-pin-invalid-error",
+ retries
+ );
+ } else if (data.error.type == "cancel") {
+ show_results_banner(
+ "error",
+ "about-webauthn-results-cancelled-by-user-error"
+ );
+ } else {
+ show_results_banner("error", "about-webauthn-results-general-error");
+ }
+ AboutWebauthnService.listen();
+ }
+ },
+
+ show_authenticator_options(options, element, l10n_base) {
+ let table = document.getElementById(element);
+ var empty_table = document.createElement("table");
+ empty_table.id = element;
+ table.parentNode.replaceChild(empty_table, table);
+ table = document.getElementById(element);
+ for (let key in options) {
+ if (key == "options") {
+ continue;
+ }
+ // Create an empty <tr> element and add it to the 1st position of the table:
+ var row = table.insertRow(0);
+
+ // Insert new cells (<td> elements) at the 1st and 2nd position of the "new" <tr> element:
+ var cell1 = row.insertCell(0);
+ var cell2 = row.insertCell(1);
+
+ // Add some text to the new cells:
+ let key_text = this._l10n.formatValueSync(
+ l10n_base + "-" + key.toLowerCase().replace(/_/g, "-")
+ );
+ var key_node = document.createTextNode(key_text);
+ cell1.appendChild(key_node);
+ var raw_value = JSON.stringify(options[key]);
+ var value = raw_value;
+ if (["true", "false", "null"].includes(raw_value)) {
+ value = this._l10n.formatValueSync(l10n_base + "-" + raw_value);
+ }
+ var value_node = document.createTextNode(value);
+ cell2.appendChild(value_node);
+ }
+ },
+
+ show_ui_based_on_authenticator_info(data) {
+ // Hide the "Please plug in a token"-message
+ document.getElementById("info-text-div").hidden = true;
+ // Show options, based on what the token supports
+ if (data.auth_info) {
+ document.getElementById("ctap2-token-info").style.display = "flex";
+ this.show_authenticator_options(
+ data.auth_info.options,
+ "authenticator-options",
+ "about-webauthn-auth-option"
+ );
+ this.show_authenticator_options(
+ data.auth_info,
+ "authenticator-info",
+ "about-webauthn-auth-info"
+ );
+ // Check if token supports PINs
+ if (data.auth_info.options.clientPin != null) {
+ document.getElementById("pin-tab-button").style.display = "flex";
+ if (data.auth_info.options.clientPin === true) {
+ // It has a Pin set
+ document.getElementById("change-pin-button").style.display = "block";
+ document.getElementById("set-pin-button").style.display = "none";
+ document.getElementById("current-pin-div").hidden = false;
+ } else {
+ // It does not have a Pin set yet
+ document.getElementById("change-pin-button").style.display = "none";
+ document.getElementById("set-pin-button").style.display = "block";
+ document.getElementById("current-pin-div").hidden = true;
+ }
+ } else {
+ document.getElementById("pin-tab-button").style.display = "none";
+ }
+
+ if (
+ data.auth_info.options.credMgmt ||
+ data.auth_info.options.credentialMgmtPreview
+ ) {
+ document.getElementById("credentials-tab-button").style.display =
+ "flex";
+ } else {
+ document.getElementById("credentials-tab-button").style.display =
+ "none";
+ }
+
+ if (
+ data.auth_info.options.bioEnroll != null ||
+ data.auth_info.options.userVerificationMgmtPreview != null
+ ) {
+ document.getElementById("bio-enrollments-tab-button").style.display =
+ "flex";
+ } else {
+ document.getElementById("bio-enrollments-tab-button").style.display =
+ "none";
+ }
+ } else {
+ // Currently auth-rs doesn't send this, because it filters out ctap2-devices.
+ // U2F / CTAP1 tokens can't be managed
+ set_info_text("about-webauthn-text-non-ctap2-device");
+ }
+ },
+
+ show_credential_list(credential_list) {
+ // We may have temporarily hidden the tab when asking the user for a PIN
+ // so we have to show it again.
+ fake_click_event_for_id("credentials-tab-button");
+ document.getElementById("credential-list-subsection").hidden = false;
+ let table = document.getElementById("credential-list");
+ var empty_table = document.createElement("table");
+ empty_table.id = "credential-list";
+ table.parentNode.replaceChild(empty_table, table);
+ if (!credential_list.length) {
+ document.getElementById("credential-list-empty-label").hidden = false;
+ return;
+ }
+ document.getElementById("credential-list-empty-label").hidden = true;
+ table = document.getElementById("credential-list");
+ credential_list.forEach(rp => {
+ // Add some text to the new cells:
+ let key_text = rp.rp.id;
+ rp.credentials.forEach(cred => {
+ let value_text = cred.user.name;
+ // Create an empty <tr> element and add it to the 1st position of the table:
+ var row = table.insertRow(0);
+ var key_node = document.createTextNode(key_text);
+ var value_node = document.createTextNode(value_text);
+ row.insertCell(0).appendChild(key_node);
+ row.insertCell(1).appendChild(value_node);
+ var delete_button = document.createElement("button");
+ delete_button.classList.add("delete-button");
+ delete_button.classList.add("credentials-button");
+ let garbage_icon = document.createElement("img");
+ garbage_icon.setAttribute(
+ "src",
+ "chrome://global/skin/icons/delete.svg"
+ );
+ garbage_icon.classList.add("delete-icon");
+ delete_button.appendChild(garbage_icon);
+ let delete_text = document.createElement("span");
+ delete_text.setAttribute(
+ "data-l10n-id",
+ "about-webauthn-delete-button"
+ );
+ delete_button.appendChild(delete_text);
+ delete_button.addEventListener("click", function () {
+ let context = document.getElementById("confirmation-context");
+ context.textContent = key_text + " - " + value_text;
+ credential_management_in_progress(true);
+ let cmd = {
+ CredentialManagement: { DeleteCredential: cred.credential_id },
+ };
+ context.setAttribute("data-ctap-command", JSON.stringify(cmd));
+ open_delete_confirmation_tab();
+ });
+ row.insertCell(2).appendChild(delete_button);
+ });
+ });
+ },
+
+ show_enrollment_list(enrollment_list) {
+ // We may have temporarily hidden the tab when asking the user for a PIN
+ // so we have to show it again.
+ fake_click_event_for_id("bio-enrollments-tab-button");
+ document.getElementById("bio-enrollment-list-subsection").hidden = false;
+ let table = document.getElementById("bio-enrollment-list");
+ var empty_table = document.createElement("table");
+ empty_table.id = "bio-enrollment-list";
+ table.parentNode.replaceChild(empty_table, table);
+ if (!enrollment_list.length) {
+ document.getElementById("bio-enrollment-list-empty-label").hidden = false;
+ return;
+ }
+ document.getElementById("bio-enrollment-list-empty-label").hidden = true;
+ table = document.getElementById("bio-enrollment-list");
+ enrollment_list.forEach(enrollment => {
+ let key_text = enrollment.template_friendly_name ?? "<unnamed>";
+ var row = table.insertRow(0);
+ var key_node = document.createTextNode(key_text);
+ row.insertCell(0).appendChild(key_node);
+ var delete_button = document.createElement("button");
+ delete_button.classList.add("delete-button");
+ delete_button.classList.add("bio-enrollment-button");
+ let garbage_icon = document.createElement("img");
+ garbage_icon.setAttribute("src", "chrome://global/skin/icons/delete.svg");
+ garbage_icon.classList.add("delete-icon");
+ delete_button.appendChild(garbage_icon);
+ let delete_text = document.createElement("span");
+ delete_text.setAttribute("data-l10n-id", "about-webauthn-delete-button");
+ delete_button.appendChild(delete_text);
+ delete_button.addEventListener("click", function () {
+ let context = document.getElementById("confirmation-context");
+ context.textContent = key_text;
+ bio_enrollment_in_progress(true);
+ let cmd = {
+ BioEnrollment: { DeleteEnrollment: enrollment.template_id },
+ };
+ context.setAttribute("data-ctap-command", JSON.stringify(cmd));
+ open_delete_confirmation_tab();
+ });
+ row.insertCell(1).appendChild(delete_button);
+ });
+ },
+};
+
+function set_info_text(l10nId) {
+ document.getElementById("info-text-div").hidden = false;
+ let field = document.getElementById("info-text-field");
+ field.setAttribute("data-l10n-id", l10nId);
+ document.getElementById("ctap2-token-info").style.display = "none";
+}
+
+function show_results_banner(result, l10n, l10n_args) {
+ let ctap_result = document.getElementById("ctap-listen-result");
+ ctap_result.setAttribute("data-l10n-id", l10n);
+ ctap_result.classList.add(result);
+ if (l10n_args) {
+ ctap_result.setAttribute("data-l10n-args", l10n_args);
+ }
+}
+
+function hide_results_banner() {
+ let res_banner = document.getElementById("ctap-listen-result");
+ let res_div = document.getElementById("ctap-listen-div");
+ let empty_banner = document.createElement("label");
+ empty_banner.id = "ctap-listen-result";
+ res_div.replaceChild(empty_banner, res_banner);
+}
+
+function operation_in_progress(name, in_progress) {
+ let buttons = Array.from(document.getElementsByClassName(name));
+ buttons.forEach(button => {
+ button.disabled = in_progress;
+ });
+}
+
+function credential_management_in_progress(in_progress) {
+ operation_in_progress("credentials-button", in_progress);
+}
+
+function bio_enrollment_in_progress(in_progress) {
+ operation_in_progress("bio-enrollment-button", in_progress);
+}
+
+function clear_bio_enrollment_samples() {
+ // Remove all previous status updates
+ let up = document.getElementById("enrollment-update");
+ while (up.firstChild) {
+ up.removeChild(up.lastChild);
+ }
+ document.getElementById("enrollment-name").value = "";
+}
+
+function fake_click_event_for_id(id) {
+ // Not using document.getElementById(id).click();
+ // here, because we have to add additional data, so we don't
+ // hide the results-div here, if there is any. 'Normal' clicking
+ // by the user will hide it.
+ const evt = new CustomEvent("click", {
+ detail: { skip_results_clearing: true },
+ });
+ document.getElementById(id).dispatchEvent(evt);
+}
+
+function reset_page() {
+ // Hide everything that needs a device to know if it should be displayed
+ document.getElementById("ctap2-token-info").style.display = "none";
+ Array.from(document.getElementsByClassName("optional-category")).forEach(
+ div => {
+ div.style.display = "none";
+ }
+ );
+
+ // Only display the "please connect a device" - text
+ set_info_text("about-webauthn-text-connect-device");
+
+ // Clear results and input fields
+ hide_results_banner();
+ var divs = Array.from(document.getElementsByTagName("input"));
+ divs.forEach(div => {
+ div.value = "";
+ });
+
+ sidebar_set_disabled(false);
+
+ // ListCredentials
+ credential_management_in_progress(false);
+ document.getElementById("credential-list-subsection").hidden = true;
+
+ // BioEnrollment
+ clear_bio_enrollment_samples();
+ document.getElementById("bio-enrollment-list-subsection").hidden = true;
+ bio_enrollment_in_progress(false);
+
+ AboutWebauthnManagerJS._previous_tab = "";
+ AboutWebauthnManagerJS._current_tab = "";
+
+ // Not using `document.getElementById("info-tab-button").click();`
+ // here, because if we were focused on a category-button that got removed (e.g.
+ // when unplugging the device), we have to reset the ARIA-related attributes
+ // first, before we can click on the button, otherwise the a11y-tests
+ // will complain. So we fake the click again.
+ const evt = {
+ detail: {},
+ currentTarget: document.getElementById("info-tab-button"),
+ };
+ open_info_tab(evt);
+}
+
+function sidebar_set_disabled(disabled) {
+ var cats = Array.from(document.getElementsByClassName("category"));
+ cats.forEach(cat => {
+ if (disabled) {
+ cat.classList.add("disabled-category");
+ } else {
+ cat.classList.remove("disabled-category");
+ }
+ });
+}
+
+function check_pin_repeat_is_correct(button) {
+ let pin = document.getElementById("new-pin");
+ let pin_repeat = document.getElementById("new-pin-repeat");
+ let has_current_pin = !document.getElementById("current-pin-div").hidden;
+ let current_pin = document.getElementById("current-pin");
+ let can_enable_button =
+ pin.value != null && pin.value != "" && pin.value == pin_repeat.value;
+ if (has_current_pin && !current_pin.value) {
+ can_enable_button = false;
+ }
+ if (!can_enable_button) {
+ pin.classList.add("different");
+ pin_repeat.classList.add("different");
+ document.getElementById("set-pin-button").disabled = true;
+ document.getElementById("change-pin-button").disabled = true;
+ return false;
+ }
+ pin.classList.remove("different");
+ pin_repeat.classList.remove("different");
+ document.getElementById("set-pin-button").disabled = false;
+ document.getElementById("change-pin-button").disabled = false;
+ return true;
+}
+
+function send_pin() {
+ close_temporary_overlay_tab();
+ let pin = document.getElementById("pin-required").value;
+ AboutWebauthnService.pinCallback(0, pin);
+}
+
+function set_pin() {
+ let pin = document.getElementById("new-pin").value;
+ let cmd = { SetPIN: pin };
+ AboutWebauthnService.runCommand(JSON.stringify(cmd));
+}
+
+function change_pin() {
+ let curr_pin = document.getElementById("current-pin").value;
+ let new_pin = document.getElementById("new-pin").value;
+ let cmd = { ChangePIN: [curr_pin, new_pin] };
+ AboutWebauthnService.runCommand(JSON.stringify(cmd));
+}
+
+function list_credentials() {
+ credential_management_in_progress(true);
+ let cmd = { CredentialManagement: "GetCredentials" };
+ AboutWebauthnService.runCommand(JSON.stringify(cmd));
+}
+
+function list_bio_enrollments() {
+ bio_enrollment_in_progress(true);
+ let cmd = { BioEnrollment: "GetEnrollments" };
+ AboutWebauthnService.runCommand(JSON.stringify(cmd));
+}
+
+function show_add_bio_enrollment_section() {
+ const evt = new CustomEvent("click", {
+ detail: { temporary_overlay: true },
+ });
+ open_tab(evt, "add-bio-enrollment-section");
+ document.getElementById("enrollment-name").focus();
+}
+
+function start_bio_enrollment() {
+ bio_enrollment_in_progress(true);
+ let name = document.getElementById("enrollment-name").value;
+ if (!name) {
+ name = null; // Empty means "Not set"
+ }
+ let cmd = { BioEnrollment: { StartNewEnrollment: name } };
+ AboutWebauthnService.runCommand(JSON.stringify(cmd));
+}
+
+function cancel_transaction() {
+ credential_management_in_progress(false);
+ bio_enrollment_in_progress(false);
+ AboutWebauthnService.cancel(0);
+}
+
+function confirm_deletion() {
+ let context = document.getElementById("confirmation-context");
+ let cmd = context.getAttribute("data-ctap-command");
+ AboutWebauthnService.runCommand(cmd);
+}
+
+function cancel_confirmation() {
+ credential_management_in_progress(false);
+ bio_enrollment_in_progress(false);
+ close_temporary_overlay_tab();
+}
+
+async function onLoad() {
+ document.getElementById("set-pin-button").addEventListener("click", set_pin);
+ document
+ .getElementById("change-pin-button")
+ .addEventListener("click", change_pin);
+ document
+ .getElementById("list-credentials-button")
+ .addEventListener("click", list_credentials);
+ document
+ .getElementById("list-bio-enrollments-button")
+ .addEventListener("click", list_bio_enrollments);
+ document
+ .getElementById("add-bio-enrollment-button")
+ .addEventListener("click", show_add_bio_enrollment_section);
+ document
+ .getElementById("start-enrollment-button")
+ .addEventListener("click", start_bio_enrollment);
+ document
+ .getElementById("new-pin")
+ .addEventListener("input", check_pin_repeat_is_correct);
+ document
+ .getElementById("new-pin-repeat")
+ .addEventListener("input", check_pin_repeat_is_correct);
+ document
+ .getElementById("current-pin")
+ .addEventListener("input", check_pin_repeat_is_correct);
+ let info_button = document.getElementById("info-tab-button");
+ info_button.addEventListener("click", open_info_tab);
+ info_button.addEventListener("keydown", handle_keydowns);
+ let pin_button = document.getElementById("pin-tab-button");
+ pin_button.addEventListener("click", open_pin_tab);
+ pin_button.addEventListener("keydown", handle_keydowns);
+ let credentials_button = document.getElementById("credentials-tab-button");
+ credentials_button.addEventListener("click", open_credentials_tab);
+ credentials_button.addEventListener("keydown", handle_keydowns);
+ let bio_enrollments_button = document.getElementById(
+ "bio-enrollments-tab-button"
+ );
+ bio_enrollments_button.addEventListener("click", open_bio_enrollments_tab);
+ bio_enrollments_button.addEventListener("keydown", handle_keydowns);
+ document
+ .getElementById("send-pin-button")
+ .addEventListener("click", send_pin);
+ document
+ .getElementById("cancel-send-pin-button")
+ .addEventListener("click", cancel_transaction);
+ document
+ .getElementById("cancel-enrollment-button")
+ .addEventListener("click", cancel_transaction);
+ document
+ .getElementById("cancel-confirmation-button")
+ .addEventListener("click", cancel_confirmation);
+ document
+ .getElementById("confirm-deletion-button")
+ .addEventListener("click", confirm_deletion);
+ AboutWebauthnManagerJS.init();
+ try {
+ AboutWebauthnService.listen();
+ } catch (ex) {
+ set_info_text("about-webauthn-text-not-available");
+ AboutWebauthnManagerJS.uninit();
+ }
+}
+
+function handle_keydowns(event) {
+ let index;
+ let event_was_handled = true;
+ let tabs = Array.from(document.getElementsByClassName("category"));
+ if (tabs.length <= 0) {
+ return;
+ }
+
+ switch (event.key) {
+ case "ArrowLeft":
+ case "ArrowUp":
+ if (event.currentTarget === tabs[0]) {
+ event.currentTarget.focus();
+ } else {
+ index = tabs.indexOf(event.currentTarget);
+ tabs[index - 1].focus();
+ }
+ break;
+
+ case "ArrowRight":
+ case "ArrowDown":
+ if (event.currentTarget === tabs[tabs.length - 1]) {
+ event.currentTarget.focus();
+ } else {
+ index = tabs.indexOf(event.currentTarget);
+ tabs[index + 1].focus();
+ }
+ break;
+
+ case "Home":
+ tabs[0].focus();
+ break;
+
+ case "End":
+ tabs[tabs.length - 1].focus();
+ break;
+
+ case "Enter":
+ case " ":
+ event.currentTarget.click();
+ break;
+
+ default:
+ event_was_handled = false;
+ break;
+ }
+
+ if (event_was_handled) {
+ event.stopPropagation();
+ event.preventDefault();
+ }
+}
+
+function open_tab(evt, tabName) {
+ var tabcontent, tablinks;
+ // Hide all others
+ tabcontent = Array.from(document.getElementsByClassName("tabcontent"));
+ tabcontent.forEach(tab => {
+ tab.style.display = "none";
+ });
+ // Display the one we selected
+ document.getElementById(tabName).style.display = "block";
+
+ // If this is a temporary overlay, like pin-required, we don't
+ // touch the sidebar and which button is selected.
+ if (!evt.detail.temporary_overlay) {
+ tablinks = Array.from(document.getElementsByClassName("category"));
+ tablinks.forEach(tablink => {
+ tablink.removeAttribute("selected");
+ tablink.setAttribute("aria-selected", "false");
+ tablink.setAttribute("tabindex", "-1");
+ tablink.disabled = false;
+ });
+ evt.currentTarget.setAttribute("selected", "true");
+ evt.currentTarget.setAttribute("tabindex", "0");
+ evt.currentTarget.setAttribute("aria-selected", "true");
+ }
+
+ if (!evt.detail.skip_results_clearing) {
+ hide_results_banner();
+ }
+ sidebar_set_disabled(false);
+ AboutWebauthnManagerJS._previous_tab = AboutWebauthnManagerJS._current_tab;
+ AboutWebauthnManagerJS._current_tab = tabName;
+}
+
+function open_info_tab(evt) {
+ open_tab(evt, "token-info-section");
+}
+function open_pin_tab(evt) {
+ open_tab(evt, "set-change-pin-section");
+}
+function open_credentials_tab(evt) {
+ open_tab(evt, "credential-management-section");
+}
+function open_bio_enrollments_tab(evt) {
+ // We can only list, if there are any registered already
+ if (
+ AboutWebauthnManagerJS._curr_data.options.bioEnroll === true ||
+ AboutWebauthnManagerJS._curr_data.options.userVerificationMgmtPreview ===
+ true
+ ) {
+ document.getElementById("list-bio-enrollments-button").style.display =
+ "inline-block";
+ } else {
+ document.getElementById("list-bio-enrollments-button").style.display =
+ "none";
+ }
+ open_tab(evt, "bio-enrollment-section");
+}
+function open_reset_tab(evt) {
+ open_tab(evt, "reset-token-section");
+}
+function open_pin_required_tab() {
+ // Remove any old value we might have had
+ document.getElementById("pin-required").value = "";
+ const evt = new CustomEvent("click", {
+ detail: {
+ temporary_overlay: true,
+ skip_results_clearing: true, // We might be called multiple times, if PIN was invalid
+ },
+ });
+ open_tab(evt, "pin-required-section");
+ document.getElementById("pin-required").focus();
+ // This is a temporary overlay, so we don't want the
+ // user to click away from it, unless via the Cancel-button.
+ sidebar_set_disabled(true);
+}
+function close_temporary_overlay_tab() {
+ const evt = new CustomEvent("click", {
+ detail: { temporary_overlay: true },
+ });
+ open_tab(evt, AboutWebauthnManagerJS._previous_tab);
+ sidebar_set_disabled(false);
+}
+function open_delete_confirmation_tab() {
+ const evt = new CustomEvent("click", {
+ detail: {
+ temporary_overlay: true,
+ },
+ });
+ open_tab(evt, "confirm-deletion-section");
+ // This is a temporary overlay, so we don't want the
+ // user to click away from it, unless via the Cancel-button.
+ sidebar_set_disabled(true);
+}
+
+try {
+ AboutWebauthnService = Cc["@mozilla.org/webauthn/service;1"].getService(
+ Ci.nsIWebAuthnService
+ );
+ document.addEventListener("DOMContentLoaded", onLoad);
+ window.addEventListener("beforeunload", event => {
+ AboutWebauthnManagerJS.uninit();
+ if (AboutWebauthnService) {
+ AboutWebauthnService.cancel(0);
+ }
+ });
+} catch (ex) {
+ // Do nothing if we fail to create a singleton instance,
+ // showing the default no-module message.
+ console.error(ex);
+}