Welcome to mirror list, hosted at ThFree Co, Russian Federation.

github.com/keepassxreboot/keepassxc-browser.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSami Vänttinen <sami.vanttinen@protonmail.com>2020-08-22 19:30:37 +0300
committerGitHub <noreply@github.com>2020-08-22 19:30:37 +0300
commitf8373b1b1acc26d9499459baa275d543e12e8028 (patch)
treef2ebb04724109187917a4c81f8cf1a42621f9160
parent9edaa267d56e24ffb40bb8376d61d308d009a37e (diff)
parent75b3120cb0c1394da0f8e461e83eb1486e44f548 (diff)
Merge pull request #961 from keepassxreboot/feature/big_refactor1.7.0-beta1
Big refactor
-rw-r--r--keepassxc-browser/_locales/en/messages.json4
-rwxr-xr-xkeepassxc-browser/background/browserAction.js93
-rwxr-xr-xkeepassxc-browser/background/event.js130
-rw-r--r--keepassxc-browser/background/init.js4
-rwxr-xr-xkeepassxc-browser/background/keepass.js92
-rwxr-xr-xkeepassxc-browser/background/page.js98
-rw-r--r--keepassxc-browser/bootstrap/bootstrap.min.css1
-rw-r--r--keepassxc-browser/content/autocomplete.js37
-rw-r--r--keepassxc-browser/content/banner.js121
-rw-r--r--keepassxc-browser/content/define.js184
-rwxr-xr-xkeepassxc-browser/content/keepassxc-browser.js2751
-rw-r--r--keepassxc-browser/content/pwgen.js132
-rw-r--r--keepassxc-browser/content/totp-field.js57
-rw-r--r--keepassxc-browser/content/ui.js28
-rw-r--r--keepassxc-browser/content/username-field.js59
-rwxr-xr-xkeepassxc-browser/global.js6
-rwxr-xr-xkeepassxc-browser/manifest.json4
-rw-r--r--keepassxc-browser/options/options.js10
-rw-r--r--keepassxc-browser/popups/popup.js2
-rw-r--r--keepassxc-browser/popups/popup_httpauth.js2
-rw-r--r--keepassxc-browser/popups/popup_login.js15
-rw-r--r--keepassxc-browser/popups/popup_multiple-fields.html39
22 files changed, 1774 insertions, 2095 deletions
diff --git a/keepassxc-browser/_locales/en/messages.json b/keepassxc-browser/_locales/en/messages.json
index 31804df..685253d 100644
--- a/keepassxc-browser/_locales/en/messages.json
+++ b/keepassxc-browser/_locales/en/messages.json
@@ -263,6 +263,10 @@
"message": "No credentials for the given username found.",
"description": "Alert message when no credentials are found for the given username."
},
+ "credentialsNoTOTPFound": {
+ "message": "No TOTP found.",
+ "description": "Alert message when no TOTP is found."
+ },
"credentialExpired": {
"message": "Expired",
"description": "If a credential is expired, this is appended to the title label."
diff --git a/keepassxc-browser/background/browserAction.js b/keepassxc-browser/background/browserAction.js
index 282cd96..bf0c26c 100755
--- a/keepassxc-browser/background/browserAction.js
+++ b/keepassxc-browser/background/browserAction.js
@@ -2,33 +2,30 @@
const browserAction = {};
-browserAction.show = function(tab) {
- let data = {};
- if (!page.tabs[tab.id] || page.tabs[tab.id].stack.length === 0) {
- browserAction.showDefault(tab);
- return;
- } else {
- data = page.tabs[tab.id].stack[page.tabs[tab.id].stack.length - 1];
+browserAction.show = function(tab, popupData) {
+ if (!popupData) {
+ popupData = page.popupData;
}
+ page.popupData = popupData;
+
browser.browserAction.setIcon({
tabId: tab.id,
- path: '/icons/toolbar/' + browserAction.generateIconName(data.iconType, data.icon)
+ path: browserAction.generateIconName(popupData.iconType)
});
- if (data.popup) {
+ if (popupData.popup) {
browser.browserAction.setPopup({
tabId: tab.id,
- popup: 'popups/' + data.popup
+ popup: `popups/${popupData.popup}.html`
});
}
};
browserAction.showDefault = async function(tab) {
- const stackData = {
- level: 1,
+ const popupData = {
iconType: 'normal',
- popup: 'popup.html'
+ popup: 'popup'
};
const response = await keepass.isConfigured().catch((err) => {
@@ -36,77 +33,41 @@ browserAction.showDefault = async function(tab) {
});
if (!response && !keepass.isKeePassXCAvailable) {
- stackData.iconType = 'cross';
+ popupData.iconType = 'cross';
} else if (keepass.isKeePassXCAvailable && keepass.isDatabaseClosed) {
- stackData.iconType = 'locked';
+ popupData.iconType = 'locked';
}
if (page.tabs[tab.id] && page.tabs[tab.id].loginList.length > 0) {
- stackData.iconType = 'questionmark';
- stackData.popup = 'popup_login.html';
+ popupData.iconType = 'questionmark';
+ popupData.popup = 'popup_login';
}
- browserAction.stackUnshift(stackData, tab.id);
- browserAction.show(tab);
+ browserAction.show(tab, popupData);
};
-browserAction.removeLevelFromStack = function(tab, level, type, dontShow) {
- if (!page.tabs[tab.id]) {
- return;
- }
-
- if (!type) {
- type = '<=';
- }
-
- const newStack = [];
- for (const i of page.tabs[tab.id].stack) {
- if ((type === '<' && i.level >= level)
- || (type === '<=' && i.level > level)
- || (type === '=' && i.level !== level)
- || (type === '==' && i.level !== level)
- || (type === '!=' && i.level === level)
- || (type === '>' && i.level <= level)
- || (type === '>=' && i.level < level)) {
- newStack.push(i);
+browserAction.updateIcon = async function(tab, iconType) {
+ if (!tab) {
+ const tabs = await browser.tabs.query({ 'active': true, 'currentWindow': true });
+ if (tabs.length === 0) {
+ return;
}
- }
-
- page.tabs[tab.id].stack = newStack;
- if (!dontShow) {
- browserAction.show(tab);
+ tab = tabs[0];
}
-};
-browserAction.stackPop = function(tabId) {
- const id = tabId || page.currentTabId;
- page.tabs[id].stack.pop();
-};
-
-browserAction.stackPush = function(data, tabId) {
- const id = tabId || page.currentTabId;
- browserAction.removeLevelFromStack({ 'id': id }, data.level, '<=', true);
- page.tabs[id].stack.push(data);
-};
-
-browserAction.stackUnshift = function(data, tabId) {
- const id = tabId || page.currentTabId;
- browserAction.removeLevelFromStack({ 'id': id }, data.level, '<=', true);
- page.tabs[id].stack.unshift(data);
+ browser.browserAction.setIcon({
+ tabId: tab.id,
+ path: browserAction.generateIconName(iconType)
+ });
};
-browserAction.generateIconName = function(iconType, icon) {
- if (icon) {
- return icon;
- }
-
+browserAction.generateIconName = function(iconType) {
let name = 'icon_';
name += (keepass.keePassXCUpdateAvailable()) ? 'new_' : '';
name += (!iconType || iconType === 'normal') ? 'normal' : iconType;
- name += '.png';
- return name;
+ return `/icons/toolbar/${name}.png`;
};
browserAction.ignoreSite = async function(url) {
diff --git a/keepassxc-browser/background/event.js b/keepassxc-browser/background/event.js
index 5b1e2f8..862f0d3 100755
--- a/keepassxc-browser/background/event.js
+++ b/keepassxc-browser/background/event.js
@@ -13,13 +13,16 @@ kpxcEvent.onMessage = async function(request, sender) {
}
};
-kpxcEvent.showStatus = async function(tab, configured) {
+kpxcEvent.showStatus = async function(tab, configured, internalPoll) {
let keyId = null;
if (configured && keepass.databaseHash !== '') {
keyId = keepass.keyRing[keepass.databaseHash].id;
}
- browserAction.showDefault(tab);
+ if (!internalPoll) {
+ browserAction.showDefault(tab);
+ }
+
const errorMessage = page.tabs[tab.id].errorMessage;
return {
identifier: keyId,
@@ -29,7 +32,7 @@ kpxcEvent.showStatus = async function(tab, configured) {
encryptionKeyUnrecognized: keepass.isEncryptionKeyUnrecognized,
associated: keepass.isAssociated(),
error: errorMessage || null,
- usernameFieldDetected: page.usernameFieldDetected
+ usernameFieldDetected: page.tabs[tab.id].usernameFieldDetected
};
};
@@ -57,11 +60,9 @@ kpxcEvent.onLoadKeyRing = async function() {
return item.keyRing;
};
-kpxcEvent.onSaveSettings = async function(tab, args = []) {
- const [ settings ] = args;
+kpxcEvent.onSaveSettings = async function(tab, settings) {
browser.storage.local.set({ 'settings': settings });
kpxcEvent.onLoadSettings(tab);
- return Promise.resolve();
};
kpxcEvent.onGetStatus = async function(tab, args = []) {
@@ -76,7 +77,7 @@ kpxcEvent.onGetStatus = async function(tab, args = []) {
}
const configured = await keepass.isConfigured();
- return kpxcEvent.showStatus(tab, configured);
+ return kpxcEvent.showStatus(tab, configured, internalPoll);
} catch (err) {
console.log('Error: No status shown: ' + err);
return Promise.reject();
@@ -107,18 +108,12 @@ kpxcEvent.lockDatabase = async function(tab) {
}
};
-kpxcEvent.onPopStack = function(tab) {
- browserAction.stackPop(tab.id);
- browserAction.show(tab);
- return Promise.resolve();
-};
-
kpxcEvent.onGetTabInformation = async function(tab) {
const id = tab.id || page.currentTabId;
return page.tabs[id];
};
-kpxcEvent.onGetConnectedDatabase = function() {
+kpxcEvent.onGetConnectedDatabase = async function() {
return Promise.resolve({
count: Object.keys(keepass.keyRing).length,
identifier: (keepass.keyRing[keepass.associated.hash]) ? keepass.keyRing[keepass.associated.hash].id : null
@@ -143,102 +138,46 @@ kpxcEvent.onUpdateAvailableKeePassXC = async function() {
return (page.settings.checkUpdateKeePassXC > 0) ? keepass.keePassXCUpdateAvailable() : false;
};
-kpxcEvent.onRemoveCredentialsFromTabInformation = function(tab) {
+kpxcEvent.onRemoveCredentialsFromTabInformation = async function(tab) {
const id = tab.id || page.currentTabId;
page.clearCredentials(id);
page.clearSubmittedCredentials();
- return Promise.resolve();
};
-kpxcEvent.onLoginPopup = function(tab, logins) {
- const stackData = {
- level: 1,
+kpxcEvent.onLoginPopup = async function(tab, logins) {
+ const popupData = {
iconType: 'questionmark',
- popup: 'popup_login.html'
+ popup: 'popup_login'
};
- browserAction.stackUnshift(stackData, tab.id);
page.tabs[tab.id].loginList = logins;
- browserAction.show(tab);
- return Promise.resolve();
+ browserAction.show(tab, popupData);
};
-kpxcEvent.initHttpAuth = function() {
+kpxcEvent.initHttpAuth = async function() {
httpAuth.init();
- return Promise.resolve();
};
-kpxcEvent.onHTTPAuthPopup = function(tab, data) {
- const stackData = {
- level: 1,
+kpxcEvent.onHTTPAuthPopup = async function(tab, data) {
+ const popupData = {
iconType: 'questionmark',
- popup: 'popup_httpauth.html'
+ popup: 'popup_httpauth'
};
- browserAction.stackUnshift(stackData, tab.id);
page.tabs[tab.id].loginList = data;
- browserAction.show(tab);
- return Promise.resolve();
-};
-
-kpxcEvent.onMultipleFieldsPopup = function(tab) {
- const stackData = {
- level: 1,
- iconType: 'normal',
- popup: 'popup_multiple-fields.html'
- };
-
- browserAction.stackUnshift(stackData, tab.id);
- browserAction.show(tab);
- return Promise.resolve();
-};
-
-kpxcEvent.pageClearLogins = function(tab, alreadyCalled) {
- if (!alreadyCalled) {
- page.clearLogins(tab.id);
- }
- return Promise.resolve();
-};
-
-kpxcEvent.pageGetLoginId = async function() {
- return page.loginId;
+ browserAction.show(tab, popupData);
};
-kpxcEvent.pageSetLoginId = function(tab, loginId) {
- page.loginId = loginId;
- return Promise.resolve();
-};
-
-kpxcEvent.pageClearSubmitted = function() {
- page.clearSubmittedCredentials();
- return Promise.resolve();
-};
-
-kpxcEvent.pageGetSubmitted = async function(tab) {
- // Do not return any credentials if the tab ID does not match.
- if (tab.id !== page.submittedCredentials.tabId) {
- return {};
- }
- return page.submittedCredentials;
-};
-
-kpxcEvent.pageSetSubmitted = function(tab, args = []) {
- const [ submitted, username, password, url, oldCredentials ] = args;
- page.setSubmittedCredentials(submitted, username, password, url, oldCredentials, tab.id);
- return Promise.resolve();
-};
-
-kpxcEvent.onUsernameFieldDetected = function(tab, detected) {
- page.usernameFieldDetected = detected;
+kpxcEvent.onUsernameFieldDetected = async function(tab, detected) {
+ page.tabs[tab.id].usernameFieldDetected = detected;
};
kpxcEvent.passwordGetFilled = async function() {
return page.passwordFilled;
};
-kpxcEvent.passwordSetFilled = function(tab, state) {
+kpxcEvent.passwordSetFilled = async function(tab, state) {
page.passwordFilled = state;
- return Promise.resolve();
};
kpxcEvent.getColorTheme = async function(tab) {
@@ -249,6 +188,12 @@ kpxcEvent.pageGetRedirectCount = async function() {
return page.redirectCount;
};
+kpxcEvent.pageClearLogins = async function(tab, alreadyCalled) {
+ if (!alreadyCalled) {
+ page.clearLogins(tab.id);
+ }
+};
+
// All methods named in this object have to be declared BEFORE this!
kpxcEvent.messageHandlers = {
'add_credentials': keepass.addCredentials,
@@ -266,26 +211,27 @@ kpxcEvent.messageHandlers = {
'get_keepassxc_versions': kpxcEvent.onGetKeePassXCVersions,
'get_status': kpxcEvent.onGetStatus,
'get_tab_information': kpxcEvent.onGetTabInformation,
+ 'get_totp': keepass.getTotp,
'init_http_auth': kpxcEvent.initHttpAuth,
'is_connected': keepass.getIsKeePassXCAvailable,
'load_keyring': kpxcEvent.onLoadKeyRing,
'load_settings': kpxcEvent.onLoadSettings,
- 'lock-database': kpxcEvent.lockDatabase,
+ 'lock_database': kpxcEvent.lockDatabase,
'page_clear_logins': kpxcEvent.pageClearLogins,
- 'page_clear_submitted': kpxcEvent.pageClearSubmitted,
- 'page_get_login_id': kpxcEvent.pageGetLoginId,
+ 'page_clear_submitted': page.clearSubmittedCredentials,
+ 'page_get_login_id': page.getLoginId,
+ 'page_get_manual_fill': page.getManualFill,
'page_get_redirect_count': kpxcEvent.pageGetRedirectCount,
- 'page_get_submitted': kpxcEvent.pageGetSubmitted,
- 'page_set_login_id': kpxcEvent.pageSetLoginId,
- 'page_set_submitted': kpxcEvent.pageSetSubmitted,
+ 'page_get_submitted': page.getSubmitted,
+ 'page_set_login_id': page.setLoginId,
+ 'page_set_manual_fill': page.setManualFill,
+ 'page_set_submitted': page.setSubmitted,
'password_get_filled': kpxcEvent.passwordGetFilled,
'password_set_filled': kpxcEvent.passwordSetFilled,
- 'pop_stack': kpxcEvent.onPopStack,
'popup_login': kpxcEvent.onLoginPopup,
- 'popup_multiple-fields': kpxcEvent.onMultipleFieldsPopup,
'reconnect': kpxcEvent.onReconnect,
'remove_credentials_from_tab_information': kpxcEvent.onRemoveCredentialsFromTabInformation,
- 'retrieve_credentials': keepass.retrieveCredentials,
+ 'retrieve_credentials': page.retrieveCredentials,
'show_default_browseraction': browserAction.showDefault,
'update_credentials': keepass.updateCredentials,
'username_field_detected': kpxcEvent.onUsernameFieldDetected,
diff --git a/keepassxc-browser/background/init.js b/keepassxc-browser/background/init.js
index 4b05ed4..7400ac8 100644
--- a/keepassxc-browser/background/init.js
+++ b/keepassxc-browser/background/init.js
@@ -8,6 +8,7 @@
await httpAuth.init();
await keepass.reconnect(null, 5000); // 5 second timeout for the first connect
await keepass.enableAutomaticReconnect();
+ await keepass.updateDatabase();
} catch (e) {
console.log('init.js failed');
}
@@ -48,9 +49,6 @@ browser.tabs.onRemoved.addListener((tabId, removeInfo) => {
* @param {object} activeInfo
*/
browser.tabs.onActivated.addListener(async function(activeInfo) {
- // Remove possible credentials from old tab information
- page.clearCredentials(page.currentTabId, true);
-
try {
const info = await browser.tabs.get(activeInfo.tabId);
if (info && info.id) {
diff --git a/keepassxc-browser/background/keepass.js b/keepassxc-browser/background/keepass.js
index 49a9ab5..a234e31 100755
--- a/keepassxc-browser/background/keepass.js
+++ b/keepassxc-browser/background/keepass.js
@@ -36,7 +36,8 @@ const kpActions = {
DATABASE_LOCKED: 'database-locked',
DATABASE_UNLOCKED: 'database-unlocked',
GET_DATABASE_GROUPS: 'get-database-groups',
- CREATE_NEW_GROUP: 'create-new-group'
+ CREATE_NEW_GROUP: 'create-new-group',
+ GET_TOTP: 'get-totp'
};
const kpErrors = {
@@ -86,6 +87,27 @@ browser.storage.local.get({ 'latestKeePassXC': { 'version': '', 'lastChecked': n
keepass.keyRing = item.keyRing;
});
+const messageBuffer = {
+ buffer: [],
+
+ addMessage(msg) {
+ if (!this.buffer.includes(msg)) {
+ this.buffer.push(msg);
+ }
+ },
+
+ matchAndRemove(msg) {
+ for (let i = 0; i < this.buffer.length; ++i) {
+ if (msg.nonce && msg.nonce === keepass.incrementedNonce(this.buffer[i].nonce)) {
+ this.buffer.splice(i, 1);
+ return true;
+ }
+ }
+
+ return false;
+ }
+};
+
keepass.sendNativeMessage = function(request, enableTimeout = false, timeoutValue) {
return new Promise((resolve, reject) => {
let timeout;
@@ -95,11 +117,16 @@ keepass.sendNativeMessage = function(request, enableTimeout = false, timeoutValu
const listener = ((port, action) => {
const handler = (msg) => {
if (msg && msg.action === action) {
- port.removeListener(handler);
- if (enableTimeout) {
- clearTimeout(timeout);
+ // Only resolve a matching response or a notification (without nonce)
+ if (!msg.nonce || messageBuffer.matchAndRemove(msg)) {
+ port.removeListener(handler);
+ if (enableTimeout) {
+ clearTimeout(timeout);
+ }
+
+ resolve(msg);
+ return;
}
- resolve(msg);
}
};
return handler;
@@ -122,6 +149,9 @@ keepass.sendNativeMessage = function(request, enableTimeout = false, timeoutValu
}, messageTimeout);
}
+ // Store the request to the buffer
+ messageBuffer.addMessage(request);
+
// Send the request
if (keepass.nativePort) {
keepass.nativePort.postMessage(request);
@@ -777,6 +807,51 @@ keepass.createNewGroup = async function(tab, args = []) {
}
};
+keepass.getTotp = async function(tab, args = []) {
+ const [ uuid, oldTotp ] = args;
+ if (!keepass.compareVersion('2.6.1', keepass.currentKeePassXC, true)) {
+ return oldTotp;
+ }
+
+ const taResponse = await keepass.testAssociation(tab, [ false ]);
+ if (!taResponse || !keepass.isConnected) {
+ return;
+ }
+
+ const kpAction = kpActions.GET_TOTP;
+ const [ nonce, incrementedNonce ] = keepass.getNonces();
+
+ const messageData = {
+ action: kpAction,
+ uuid: uuid
+ };
+
+ try {
+ const request = keepass.buildRequest(kpAction, keepass.encrypt(messageData, nonce), nonce, keepass.clientID);
+ const response = await keepass.sendNativeMessage(request);
+ if (response.message && response.nonce) {
+ const res = keepass.decrypt(response.message, response.nonce);
+ if (!res) {
+ keepass.handleError(tab, kpErrors.CANNOT_DECRYPT_MESSAGE);
+ return;
+ }
+
+ const message = nacl.util.encodeUTF8(res);
+ const parsed = JSON.parse(message);
+ if (keepass.verifyResponse(parsed, incrementedNonce) && parsed.totp) {
+ keepass.updateLastUsed(keepass.databaseHash);
+ return parsed.totp;
+ }
+ } else if (response.error && response.errorCode) {
+ keepass.handleError(tab, response.errorCode, response.error);
+ }
+
+ return;
+ } catch (err) {
+ console.log('getTotp failed: ', err);
+ }
+};
+
keepass.generateNewKeyPair = function() {
keepass.keyPair = nacl.box.keyPair();
//console.log(nacl.util.encodeBase64(keepass.keyPair.publicKey) + ' ' + nacl.util.encodeBase64(keepass.keyPair.secretKey));
@@ -1159,9 +1234,7 @@ keepass.reconnect = async function(tab, connectionTimeout) {
keepass.updatePopup = function(iconType) {
if (page && page.tabs.length > 0) {
- const data = page.tabs[page.currentTabId].stack[page.tabs[page.currentTabId].stack.length - 1];
- data.iconType = iconType;
- browserAction.show({ 'id': page.currentTabId });
+ browserAction.updateIcon(undefined, iconType);
}
};
@@ -1169,7 +1242,8 @@ keepass.updatePopup = function(iconType) {
keepass.updateDatabase = async function() {
keepass.associated.value = false;
keepass.associated.hash = null;
- await keepass.testAssociation(null);
+ page.clearAllLogins();
+ await keepass.testAssociation(null, [ true ]);
const configured = await keepass.isConfigured();
keepass.updatePopup(configured ? 'normal' : 'locked');
keepass.updateDatabaseHashToContent();
diff --git a/keepassxc-browser/background/page.js b/keepassxc-browser/background/page.js
index d75af94..0099b1e 100755
--- a/keepassxc-browser/background/page.js
+++ b/keepassxc-browser/background/page.js
@@ -24,14 +24,20 @@ const defaultSettings = {
var page = {};
page.blockedTabs = [];
+page.currentRequest = {};
page.currentTabId = -1;
page.loginId = -1;
+page.manualFill = ManualFill.NONE;
page.passwordFilled = false;
page.redirectCount = 0;
page.submitted = false;
page.submittedCredentials = {};
page.tabs = [];
-page.usernameFieldDetected = false;
+
+page.popupData = {
+ iconType: 'normal',
+ popup: 'popup'
+};
page.initSettings = async function() {
try {
@@ -136,11 +142,11 @@ page.initOpenedTabs = async function() {
// Set initial tab-ID
const currentTabs = await browser.tabs.query({ active: true, currentWindow: true });
if (currentTabs.length === 0) {
- return Promise.resolve();
+ return;
}
+
page.currentTabId = currentTabs[0].id;
- browserAction.show(currentTabs[0]);
- return Promise.resolve();
+ browserAction.showDefault(currentTabs[0]);
} catch (err) {
console.log('page.initOpenedTabs error: ' + err);
return Promise.reject();
@@ -149,18 +155,18 @@ page.initOpenedTabs = async function() {
page.switchTab = function(tab) {
browserAction.showDefault(tab);
- browser.tabs.sendMessage(tab.id, { action: 'activated_tab' }).catch((e) => {});
+ browser.tabs.sendMessage(tab.id, { action: 'activated_tab' }).catch((e) => {
+ console.log('Cannot send activated_tab message: ', e);
+ });
};
-page.clearCredentials = function(tabId, complete) {
+page.clearCredentials = async function(tabId, complete) {
if (!page.tabs[tabId]) {
return;
}
- page.usernameFieldDetected = false;
page.passwordFilled = false;
page.tabs[tabId].credentials = [];
- delete page.tabs[tabId].credentials;
if (complete) {
page.clearLogins(tabId);
@@ -176,10 +182,19 @@ page.clearLogins = function(tabId) {
return;
}
+ page.tabs[tabId].credentials = [];
page.tabs[tabId].loginList = [];
+ page.currentRequest = {};
page.passwordFilled = false;
};
+// Clear all logins from all pages and update the content scripts
+page.clearAllLogins = function() {
+ for (const tabId of Object.keys(page.tabs)) {
+ page.clearCredentials(Number(tabId), true);
+ }
+};
+
page.setSubmittedCredentials = function(submitted, username, password, url, oldCredentials, tabId) {
page.submittedCredentials.submitted = submitted;
page.submittedCredentials.username = username;
@@ -189,17 +204,18 @@ page.setSubmittedCredentials = function(submitted, username, password, url, oldC
page.submittedCredentials.tabId = tabId;
};
-page.clearSubmittedCredentials = function() {
+page.clearSubmittedCredentials = async function() {
page.submitted = false;
page.submittedCredentials = {};
};
page.createTabEntry = function(tabId) {
page.tabs[tabId] = {
- 'stack': [],
- 'errorMessage': null,
- 'loginList': []
+ credentials: [],
+ errorMessage: null,
+ loginList: []
};
+
page.clearSubmittedCredentials();
};
@@ -221,3 +237,61 @@ page.removePageInformationFromNotExistingTabs = async function() {
}
}
};
+
+// Retrieves the credentials. Returns cached values when found.
+// Page reload or tab switch clears the cache.
+page.retrieveCredentials = async function(tab, args = []) {
+ const [ url, submitUrl ] = args;
+ if (page.tabs[tab.id] && page.tabs[tab.id].credentials.length > 0) {
+ return page.tabs[tab.id].credentials;
+ }
+
+ // Ignore duplicate requests
+ if (page.currentRequest.url === url && page.currentRequest.submitUrl === submitUrl) {
+ return [];
+ } else {
+ page.currentRequest.url = url;
+ page.currentRequest.submitUrl = submitUrl;
+ }
+
+ const credentials = await keepass.retrieveCredentials(tab, args);
+ page.tabs[tab.id].credentials = credentials;
+ return credentials;
+};
+
+page.getLoginId = async function(tab) {
+ // If there's only one credential available and loginId is not set
+ if (page.loginId < 0
+ && page.tabs[tab.id]
+ && page.tabs[tab.id].credentials.length === 1) {
+ return 0; // Index to the first credential
+ }
+
+ return page.loginId;
+};
+
+page.setLoginId = async function(tab, loginId) {
+ page.loginId = loginId;
+};
+
+page.getManualFill = async function(tab) {
+ return page.manualFill;
+};
+
+page.setManualFill = async function(tab, manualFill) {
+ page.manualFill = manualFill;
+};
+
+page.getSubmitted = async function(tab) {
+ // Do not return any credentials if the tab ID does not match.
+ if (tab.id !== page.submittedCredentials.tabId) {
+ return {};
+ }
+
+ return page.submittedCredentials;
+};
+
+page.setSubmitted = async function(tab, args = []) {
+ const [ submitted, username, password, url, oldCredentials ] = args;
+ page.setSubmittedCredentials(submitted, username, password, url, oldCredentials, tab.id);
+};
diff --git a/keepassxc-browser/bootstrap/bootstrap.min.css b/keepassxc-browser/bootstrap/bootstrap.min.css
index 86b6845..613d28a 100644
--- a/keepassxc-browser/bootstrap/bootstrap.min.css
+++ b/keepassxc-browser/bootstrap/bootstrap.min.css
@@ -4,4 +4,3 @@
* Copyright 2011-2019 Twitter, Inc.
* Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE)
*/:root{--blue:#007bff;--indigo:#6610f2;--purple:#6f42c1;--pink:#e83e8c;--red:#dc3545;--orange:#fd7e14;--yellow:#ffc107;--green:#28a745;--teal:#20c997;--cyan:#17a2b8;--white:#fff;--gray:#6c757d;--gray-dark:#343a40;--primary:#007bff;--secondary:#6c757d;--success:#28a745;--info:#17a2b8;--warning:#ffc107;--danger:#dc3545;--light:#f8f9fa;--dark:#343a40;--breakpoint-xs:0;--breakpoint-sm:576px;--breakpoint-md:768px;--breakpoint-lg:992px;--breakpoint-xl:1200px;--font-family-sans-serif:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--font-family-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace}*,::after,::before{box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}article,aside,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-size:1rem;font-weight:400;line-height:1.5;color:#212529;text-align:left;background-color:#fff}[tabindex="-1"]:focus:not(:focus-visible){outline:0!important}hr{box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem}p{margin-top:0;margin-bottom:1rem}abbr[data-original-title],abbr[title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;border-bottom:0;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#007bff;text-decoration:none;background-color:transparent}a:hover{color:#0056b3;text-decoration:underline}a:not([href]){color:inherit;text-decoration:none}a:not([href]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em}pre{margin-top:0;margin-bottom:1rem;overflow:auto}figure{margin:0 0 1rem}img{vertical-align:middle;border-style:none}svg{overflow:hidden;vertical-align:middle}table{border-collapse:collapse}caption{padding-top:.75rem;padding-bottom:.75rem;color:#6c757d;text-align:left;caption-side:bottom}th{text-align:inherit}label{display:inline-block;margin-bottom:.5rem}button{border-radius:0}button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}select{word-wrap:normal}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{padding:0;border-style:none}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=date],input[type=datetime-local],input[type=month],input[type=time]{-webkit-appearance:listbox}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;max-width:100%;padding:0;margin-bottom:.5rem;font-size:1.5rem;line-height:inherit;color:inherit;white-space:normal}progress{vertical-align:baseline}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:none}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item;cursor:pointer}template{display:none}[hidden]{display:none!important}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-bottom:.5rem;font-weight:500;line-height:1.2}.h1,h1{font-size:2.5rem}.h2,h2{font-size:2rem}.h3,h3{font-size:1.75rem}.h4,h4{font-size:1.5rem}.h5,h5{font-size:1.25rem}.h6,h6{font-size:1rem}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:6rem;font-weight:300;line-height:1.2}.display-2{font-size:5.5rem;font-weight:300;line-height:1.2}.display-3{font-size:4.5rem;font-weight:300;line-height:1.2}.display-4{font-size:3.5rem;font-weight:300;line-height:1.2}hr{margin-top:1rem;margin-bottom:1rem;border:0;border-top:1px solid rgba(0,0,0,.1)}.small,small{font-size:80%;font-weight:400}.mark,mark{padding:.2em;background-color:#fcf8e3}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:90%;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote-footer{display:block;font-size:80%;color:#6c757d}.blockquote-footer::before{content:"\2014\00A0"}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:#fff;border:1px solid #dee2e6;border-radius:.25rem;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:90%;color:#6c757d}code{font-size:87.5%;color:#e83e8c;word-wrap:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:87.5%;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:100%;font-weight:700}pre{display:block;font-size:87.5%;color:#212529}pre code{font-size:inherit;color:inherit;word-break:normal}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:576px){.container{max-width:540px}}@media (min-width:768px){.container{max-width:720px}}@media (min-width:992px){.container{max-width:960px}}@media (min-width:1200px){.container{max-width:1140px}}.container-fluid,.container-lg,.container-md,.container-sm,.container-xl{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:576px){.container,.container-sm{max-width:540px}}@media (min-width:768px){.container,.container-md,.container-sm{max-width:720px}}@media (min-width:992px){.container,.container-lg,.container-md,.container-sm{max-width:960px}}@media (min-width:1200px){.container,.container-lg,.container-md,.container-sm,.container-xl{max-width:1140px}}.row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-15px;margin-left:-15px}.no-gutters{margin-right:0;margin-left:0}.no-gutters>.col,.no-gutters>[class*=col-]{padding-right:0;padding-left:0}.col,.col-1,.col-10,.col-11,.col-12,.col-2,.col-3,.col-4,.col-5,.col-6,.col-7,.col-8,.col-9,.col-auto,.col-lg,.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-auto,.col-md,.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-auto,.col-sm,.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-auto,.col-xl,.col-xl-1,.col-xl-10,.col-xl-11,.col-xl-12,.col-xl-2,.col-xl-3,.col-xl-4,.col-xl-5,.col-xl-6,.col-xl-7,.col-xl-8,.col-xl-9,.col-xl-auto{position:relative;width:100%;padding-right:15px;padding-left:15px}.col{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-first{-ms-flex-order:-1;order:-1}.order-last{-ms-flex-order:13;order:13}.order-0{-ms-flex-order:0;order:0}.order-1{-ms-flex-order:1;order:1}.order-2{-ms-flex-order:2;order:2}.order-3{-ms-flex-order:3;order:3}.order-4{-ms-flex-order:4;order:4}.order-5{-ms-flex-order:5;order:5}.order-6{-ms-flex-order:6;order:6}.order-7{-ms-flex-order:7;order:7}.order-8{-ms-flex-order:8;order:8}.order-9{-ms-flex-order:9;order:9}.order-10{-ms-flex-order:10;order:10}.order-11{-ms-flex-order:11;order:11}.order-12{-ms-flex-order:12;order:12}.offset-1{margin-left:8.333333%}.offset-2{margin-left:16.666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.333333%}.offset-5{margin-left:41.666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.333333%}.offset-8{margin-left:66.666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.333333%}.offset-11{margin-left:91.666667%}@media (min-width:576px){.col-sm{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-sm-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-sm-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-sm-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-sm-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-sm-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-sm-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-sm-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-sm-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-sm-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-sm-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-sm-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-sm-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-sm-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-sm-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-sm-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-sm-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-sm-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-sm-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-sm-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-sm-first{-ms-flex-order:-1;order:-1}.order-sm-last{-ms-flex-order:13;order:13}.order-sm-0{-ms-flex-order:0;order:0}.order-sm-1{-ms-flex-order:1;order:1}.order-sm-2{-ms-flex-order:2;order:2}.order-sm-3{-ms-flex-order:3;order:3}.order-sm-4{-ms-flex-order:4;order:4}.order-sm-5{-ms-flex-order:5;order:5}.order-sm-6{-ms-flex-order:6;order:6}.order-sm-7{-ms-flex-order:7;order:7}.order-sm-8{-ms-flex-order:8;order:8}.order-sm-9{-ms-flex-order:9;order:9}.order-sm-10{-ms-flex-order:10;order:10}.order-sm-11{-ms-flex-order:11;order:11}.order-sm-12{-ms-flex-order:12;order:12}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.333333%}.offset-sm-2{margin-left:16.666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.333333%}.offset-sm-5{margin-left:41.666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.333333%}.offset-sm-8{margin-left:66.666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.333333%}.offset-sm-11{margin-left:91.666667%}}@media (min-width:768px){.col-md{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-md-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-md-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-md-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-md-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-md-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-md-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-md-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-md-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-md-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-md-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-md-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-md-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-md-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-md-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-md-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-md-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-md-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-md-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-md-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-md-first{-ms-flex-order:-1;order:-1}.order-md-last{-ms-flex-order:13;order:13}.order-md-0{-ms-flex-order:0;order:0}.order-md-1{-ms-flex-order:1;order:1}.order-md-2{-ms-flex-order:2;order:2}.order-md-3{-ms-flex-order:3;order:3}.order-md-4{-ms-flex-order:4;order:4}.order-md-5{-ms-flex-order:5;order:5}.order-md-6{-ms-flex-order:6;order:6}.order-md-7{-ms-flex-order:7;order:7}.order-md-8{-ms-flex-order:8;order:8}.order-md-9{-ms-flex-order:9;order:9}.order-md-10{-ms-flex-order:10;order:10}.order-md-11{-ms-flex-order:11;order:11}.order-md-12{-ms-flex-order:12;order:12}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.333333%}.offset-md-2{margin-left:16.666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.333333%}.offset-md-5{margin-left:41.666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.333333%}.offset-md-8{margin-left:66.666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.333333%}.offset-md-11{margin-left:91.666667%}}@media (min-width:992px){.col-lg{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-lg-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-lg-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-lg-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-lg-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-lg-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-lg-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-lg-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-lg-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-lg-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-lg-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-lg-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-lg-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-lg-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-lg-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-lg-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-lg-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-lg-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-lg-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-lg-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-lg-first{-ms-flex-order:-1;order:-1}.order-lg-last{-ms-flex-order:13;order:13}.order-lg-0{-ms-flex-order:0;order:0}.order-lg-1{-ms-flex-order:1;order:1}.order-lg-2{-ms-flex-order:2;order:2}.order-lg-3{-ms-flex-order:3;order:3}.order-lg-4{-ms-flex-order:4;order:4}.order-lg-5{-ms-flex-order:5;order:5}.order-lg-6{-ms-flex-order:6;order:6}.order-lg-7{-ms-flex-order:7;order:7}.order-lg-8{-ms-flex-order:8;order:8}.order-lg-9{-ms-flex-order:9;order:9}.order-lg-10{-ms-flex-order:10;order:10}.order-lg-11{-ms-flex-order:11;order:11}.order-lg-12{-ms-flex-order:12;order:12}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.333333%}.offset-lg-2{margin-left:16.666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.333333%}.offset-lg-5{margin-left:41.666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.333333%}.offset-lg-8{margin-left:66.666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.333333%}.offset-lg-11{margin-left:91.666667%}}@media (min-width:1200px){.col-xl{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-xl-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-xl-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-xl-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-xl-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-xl-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-xl-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-xl-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-xl-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-xl-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-xl-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-xl-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-xl-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-xl-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-xl-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-xl-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-xl-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-xl-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-xl-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-xl-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-xl-first{-ms-flex-order:-1;order:-1}.order-xl-last{-ms-flex-order:13;order:13}.order-xl-0{-ms-flex-order:0;order:0}.order-xl-1{-ms-flex-order:1;order:1}.order-xl-2{-ms-flex-order:2;order:2}.order-xl-3{-ms-flex-order:3;order:3}.order-xl-4{-ms-flex-order:4;order:4}.order-xl-5{-ms-flex-order:5;order:5}.order-xl-6{-ms-flex-order:6;order:6}.order-xl-7{-ms-flex-order:7;order:7}.order-xl-8{-ms-flex-order:8;order:8}.order-xl-9{-ms-flex-order:9;order:9}.order-xl-10{-ms-flex-order:10;order:10}.order-xl-11{-ms-flex-order:11;order:11}.order-xl-12{-ms-flex-order:12;order:12}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.333333%}.offset-xl-2{margin-left:16.666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.333333%}.offset-xl-5{margin-left:41.666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.333333%}.offset-xl-8{margin-left:66.666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.333333%}.offset-xl-11{margin-left:91.666667%}}.table{width:100%;margin-bottom:1rem;color:#212529}.table td,.table th{padding:.75rem;vertical-align:top;border-top:1px solid #dee2e6}.table thead th{vertical-align:bottom;border-bottom:2px solid #dee2e6}.table tbody+tbody{border-top:2px solid #dee2e6}.table-sm td,.table-sm th{padding:.3rem}.table-bordered{border:1px solid #dee2e6}.table-bordered td,.table-bordered th{border:1px solid #dee2e6}.table-bordered thead td,.table-bordered thead th{border-bottom-width:2px}.table-borderless tbody+tbody,.table-borderless td,.table-borderless th,.table-borderless thead th{border:0}.table-striped tbody tr:nth-of-type(odd){background-color:rgba(0,0,0,.05)}.table-hover tbody tr:hover{color:#212529;background-color:rgba(0,0,0,.075)}.table-primary,.table-primary>td,.table-primary>th{background-color:#b8daff}.table-primary tbody+tbody,.table-primary td,.table-primary th,.table-primary thead th{border-color:#7abaff}.table-hover .table-primary:hover{background-color:#9fcdff}.table-hover .table-primary:hover>td,.table-hover .table-primary:hover>th{background-color:#9fcdff}.table-secondary,.table-secondary>td,.table-secondary>th{background-color:#d6d8db}.table-secondary tbody+tbody,.table-secondary td,.table-secondary th,.table-secondary thead th{border-color:#b3b7bb}.table-hover .table-secondary:hover{background-color:#c8cbcf}.table-hover .table-secondary:hover>td,.table-hover .table-secondary:hover>th{background-color:#c8cbcf}.table-success,.table-success>td,.table-success>th{background-color:#c3e6cb}.table-success tbody+tbody,.table-success td,.table-success th,.table-success thead th{border-color:#8fd19e}.table-hover .table-success:hover{background-color:#b1dfbb}.table-hover .table-success:hover>td,.table-hover .table-success:hover>th{background-color:#b1dfbb}.table-info,.table-info>td,.table-info>th{background-color:#bee5eb}.table-info tbody+tbody,.table-info td,.table-info th,.table-info thead th{border-color:#86cfda}.table-hover .table-info:hover{background-color:#abdde5}.table-hover .table-info:hover>td,.table-hover .table-info:hover>th{background-color:#abdde5}.table-warning,.table-warning>td,.table-warning>th{background-color:#ffeeba}.table-warning tbody+tbody,.table-warning td,.table-warning th,.table-warning thead th{border-color:#ffdf7e}.table-hover .table-warning:hover{background-color:#ffe8a1}.table-hover .table-warning:hover>td,.table-hover .table-warning:hover>th{background-color:#ffe8a1}.table-danger,.table-danger>td,.table-danger>th{background-color:#f5c6cb}.table-danger tbody+tbody,.table-danger td,.table-danger th,.table-danger thead th{border-color:#ed969e}.table-hover .table-danger:hover{background-color:#f1b0b7}.table-hover .table-danger:hover>td,.table-hover .table-danger:hover>th{background-color:#f1b0b7}.table-light,.table-light>td,.table-light>th{background-color:#fdfdfe}.table-light tbody+tbody,.table-light td,.table-light th,.table-light thead th{border-color:#fbfcfc}.table-hover .table-light:hover{background-color:#ececf6}.table-hover .table-light:hover>td,.table-hover .table-light:hover>th{background-color:#ececf6}.table-dark,.table-dark>td,.table-dark>th{background-color:#c6c8ca}.table-dark tbody+tbody,.table-dark td,.table-dark th,.table-dark thead th{border-color:#95999c}.table-hover .table-dark:hover{background-color:#b9bbbe}.table-hover .table-dark:hover>td,.table-hover .table-dark:hover>th{background-color:#b9bbbe}.table-active,.table-active>td,.table-active>th{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover>td,.table-hover .table-active:hover>th{background-color:rgba(0,0,0,.075)}.table .thead-dark th{color:#fff;background-color:#343a40;border-color:#454d55}.table .thead-light th{color:#495057;background-color:#e9ecef;border-color:#dee2e6}.table-dark{color:#fff;background-color:#343a40}.table-dark td,.table-dark th,.table-dark thead th{border-color:#454d55}.table-dark.table-bordered{border:0}.table-dark.table-striped tbody tr:nth-of-type(odd){background-color:rgba(255,255,255,.05)}.table-dark.table-hover tbody tr:hover{color:#fff;background-color:rgba(255,255,255,.075)}@media (max-width:575.98px){.table-responsive-sm{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-sm>.table-bordered{border:0}}@media (max-width:767.98px){.table-responsive-md{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-md>.table-bordered{border:0}}@media (max-width:991.98px){.table-responsive-lg{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-lg>.table-bordered{border:0}}@media (max-width:1199.98px){.table-responsive-xl{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-xl>.table-bordered{border:0}}.table-responsive{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive>.table-bordered{border:0}.form-control{display:block;width:100%;height:calc(1.5em + .75rem + 2px);padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;background-color:#fff;background-clip:padding-box;border:1px solid #ced4da;border-radius:.25rem;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control{transition:none}}.form-control::-ms-expand{background-color:transparent;border:0}.form-control:-moz-focusring{color:transparent;text-shadow:0 0 0 #495057}.form-control:focus{color:#495057;background-color:#fff;border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.form-control::-webkit-input-placeholder{color:#6c757d;opacity:1}.form-control::-moz-placeholder{color:#6c757d;opacity:1}.form-control:-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::placeholder{color:#6c757d;opacity:1}.form-control:disabled,.form-control[readonly]{background-color:#e9ecef;opacity:1}select.form-control:focus::-ms-value{color:#495057;background-color:#fff}.form-control-file,.form-control-range{display:block;width:100%}.col-form-label{padding-top:calc(.375rem + 1px);padding-bottom:calc(.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + 1px);padding-bottom:calc(.5rem + 1px);font-size:1.25rem;line-height:1.5}.col-form-label-sm{padding-top:calc(.25rem + 1px);padding-bottom:calc(.25rem + 1px);font-size:.875rem;line-height:1.5}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;font-size:1rem;line-height:1.5;color:#212529;background-color:transparent;border:solid transparent;border-width:1px 0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm{padding-right:0;padding-left:0}.form-control-sm{height:calc(1.5em + .5rem + 2px);padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.form-control-lg{height:calc(1.5em + 1rem + 2px);padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}select.form-control[multiple],select.form-control[size]{height:auto}textarea.form-control{height:auto}.form-group{margin-bottom:1rem}.form-text{display:block;margin-top:.25rem}.form-row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-5px;margin-left:-5px}.form-row>.col,.form-row>[class*=col-]{padding-right:5px;padding-left:5px}.form-check{position:relative;display:block;padding-left:1.25rem}.form-check-input{position:absolute;margin-top:.3rem;margin-left:-1.25rem}.form-check-input:disabled~.form-check-label,.form-check-input[disabled]~.form-check-label{color:#6c757d}.form-check-label{margin-bottom:0}.form-check-inline{display:-ms-inline-flexbox;display:inline-flex;-ms-flex-align:center;align-items:center;padding-left:0;margin-right:.75rem}.form-check-inline .form-check-input{position:static;margin-top:0;margin-right:.3125rem;margin-left:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#28a745}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;line-height:1.5;color:#fff;background-color:rgba(40,167,69,.9);border-radius:.25rem}.is-valid~.valid-feedback,.is-valid~.valid-tooltip,.was-validated :valid~.valid-feedback,.was-validated :valid~.valid-tooltip{display:block}.form-control.is-valid,.was-validated .form-control:valid{border-color:#28a745;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%2328a745' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-valid:focus,.was-validated .form-control:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.custom-select.is-valid,.was-validated .custom-select:valid{border-color:#28a745;padding-right:calc(.75em + 2.3125rem);background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%2328a745' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e") #fff no-repeat center right 1.75rem/calc(.75em + .375rem) calc(.75em + .375rem)}.custom-select.is-valid:focus,.was-validated .custom-select:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:#28a745}.form-check-input.is-valid~.valid-feedback,.form-check-input.is-valid~.valid-tooltip,.was-validated .form-check-input:valid~.valid-feedback,.was-validated .form-check-input:valid~.valid-tooltip{display:block}.custom-control-input.is-valid~.custom-control-label,.was-validated .custom-control-input:valid~.custom-control-label{color:#28a745}.custom-control-input.is-valid~.custom-control-label::before,.was-validated .custom-control-input:valid~.custom-control-label::before{border-color:#28a745}.custom-control-input.is-valid:checked~.custom-control-label::before,.was-validated .custom-control-input:valid:checked~.custom-control-label::before{border-color:#34ce57;background-color:#34ce57}.custom-control-input.is-valid:focus~.custom-control-label::before,.was-validated .custom-control-input:valid:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.custom-control-input.is-valid:focus:not(:checked)~.custom-control-label::before,.was-validated .custom-control-input:valid:focus:not(:checked)~.custom-control-label::before{border-color:#28a745}.custom-file-input.is-valid~.custom-file-label,.was-validated .custom-file-input:valid~.custom-file-label{border-color:#28a745}.custom-file-input.is-valid:focus~.custom-file-label,.was-validated .custom-file-input:valid:focus~.custom-file-label{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#dc3545}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;line-height:1.5;color:#fff;background-color:rgba(220,53,69,.9);border-radius:.25rem}.is-invalid~.invalid-feedback,.is-invalid~.invalid-tooltip,.was-validated :invalid~.invalid-feedback,.was-validated :invalid~.invalid-tooltip{display:block}.form-control.is-invalid,.was-validated .form-control:invalid{border-color:#dc3545;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' fill='none' stroke='%23dc3545' viewBox='0 0 12 12'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-invalid:focus,.was-validated .form-control:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.custom-select.is-invalid,.was-validated .custom-select:invalid{border-color:#dc3545;padding-right:calc(.75em + 2.3125rem);background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' fill='none' stroke='%23dc3545' viewBox='0 0 12 12'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e") #fff no-repeat center right 1.75rem/calc(.75em + .375rem) calc(.75em + .375rem)}.custom-select.is-invalid:focus,.was-validated .custom-select:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:#dc3545}.form-check-input.is-invalid~.invalid-feedback,.form-check-input.is-invalid~.invalid-tooltip,.was-validated .form-check-input:invalid~.invalid-feedback,.was-validated .form-check-input:invalid~.invalid-tooltip{display:block}.custom-control-input.is-invalid~.custom-control-label,.was-validated .custom-control-input:invalid~.custom-control-label{color:#dc3545}.custom-control-input.is-invalid~.custom-control-label::before,.was-validated .custom-control-input:invalid~.custom-control-label::before{border-color:#dc3545}.custom-control-input.is-invalid:checked~.custom-control-label::before,.was-validated .custom-control-input:invalid:checked~.custom-control-label::before{border-color:#e4606d;background-color:#e4606d}.custom-control-input.is-invalid:focus~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.custom-control-input.is-invalid:focus:not(:checked)~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus:not(:checked)~.custom-control-label::before{border-color:#dc3545}.custom-file-input.is-invalid~.custom-file-label,.was-validated .custom-file-input:invalid~.custom-file-label{border-color:#dc3545}.custom-file-input.is-invalid:focus~.custom-file-label,.was-validated .custom-file-input:invalid:focus~.custom-file-label{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-inline{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center}.form-inline .form-check{width:100%}@media (min-width:576px){.form-inline label{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;margin-bottom:0}.form-inline .form-group{display:-ms-flexbox;display:flex;-ms-flex:0 0 auto;flex:0 0 auto;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center;margin-bottom:0}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-plaintext{display:inline-block}.form-inline .custom-select,.form-inline .input-group{width:auto}.form-inline .form-check{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:auto;padding-left:0}.form-inline .form-check-input{position:relative;-ms-flex-negative:0;flex-shrink:0;margin-top:0;margin-right:.25rem;margin-left:0}.form-inline .custom-control{-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center}.form-inline .custom-control-label{margin-bottom:0}}.btn{display:inline-block;font-weight:400;color:#212529;text-align:center;vertical-align:middle;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:transparent;border:1px solid transparent;padding:.375rem .75rem;font-size:1rem;line-height:1.5;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.btn{transition:none}}.btn:hover{color:#212529;text-decoration:none}.btn.focus,.btn:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.btn.disabled,.btn:disabled{opacity:.65}a.btn.disabled,fieldset:disabled a.btn{pointer-events:none}.btn-primary{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:hover{color:#fff;background-color:#0069d9;border-color:#0062cc}.btn-primary.focus,.btn-primary:focus{color:#fff;background-color:#0069d9;border-color:#0062cc;box-shadow:0 0 0 .2rem rgba(38,143,255,.5)}.btn-primary.disabled,.btn-primary:disabled{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:not(:disabled):not(.disabled).active,.btn-primary:not(:disabled):not(.disabled):active,.show>.btn-primary.dropdown-toggle{color:#fff;background-color:#0062cc;border-color:#005cbf}.btn-primary:not(:disabled):not(.disabled).active:focus,.btn-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(38,143,255,.5)}.btn-secondary{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:hover{color:#fff;background-color:#5a6268;border-color:#545b62}.btn-secondary.focus,.btn-secondary:focus{color:#fff;background-color:#5a6268;border-color:#545b62;box-shadow:0 0 0 .2rem rgba(130,138,145,.5)}.btn-secondary.disabled,.btn-secondary:disabled{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:not(:disabled):not(.disabled).active,.btn-secondary:not(:disabled):not(.disabled):active,.show>.btn-secondary.dropdown-toggle{color:#fff;background-color:#545b62;border-color:#4e555b}.btn-secondary:not(:disabled):not(.disabled).active:focus,.btn-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(130,138,145,.5)}.btn-success{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:hover{color:#fff;background-color:#218838;border-color:#1e7e34}.btn-success.focus,.btn-success:focus{color:#fff;background-color:#218838;border-color:#1e7e34;box-shadow:0 0 0 .2rem rgba(72,180,97,.5)}.btn-success.disabled,.btn-success:disabled{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:not(:disabled):not(.disabled).active,.btn-success:not(:disabled):not(.disabled):active,.show>.btn-success.dropdown-toggle{color:#fff;background-color:#1e7e34;border-color:#1c7430}.btn-success:not(:disabled):not(.disabled).active:focus,.btn-success:not(:disabled):not(.disabled):active:focus,.show>.btn-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(72,180,97,.5)}.btn-info{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:hover{color:#fff;background-color:#138496;border-color:#117a8b}.btn-info.focus,.btn-info:focus{color:#fff;background-color:#138496;border-color:#117a8b;box-shadow:0 0 0 .2rem rgba(58,176,195,.5)}.btn-info.disabled,.btn-info:disabled{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:not(:disabled):not(.disabled).active,.btn-info:not(:disabled):not(.disabled):active,.show>.btn-info.dropdown-toggle{color:#fff;background-color:#117a8b;border-color:#10707f}.btn-info:not(:disabled):not(.disabled).active:focus,.btn-info:not(:disabled):not(.disabled):active:focus,.show>.btn-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(58,176,195,.5)}.btn-warning{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:hover{color:#212529;background-color:#e0a800;border-color:#d39e00}.btn-warning.focus,.btn-warning:focus{color:#212529;background-color:#e0a800;border-color:#d39e00;box-shadow:0 0 0 .2rem rgba(222,170,12,.5)}.btn-warning.disabled,.btn-warning:disabled{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:not(:disabled):not(.disabled).active,.btn-warning:not(:disabled):not(.disabled):active,.show>.btn-warning.dropdown-toggle{color:#212529;background-color:#d39e00;border-color:#c69500}.btn-warning:not(:disabled):not(.disabled).active:focus,.btn-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(222,170,12,.5)}.btn-danger{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:hover{color:#fff;background-color:#c82333;border-color:#bd2130}.btn-danger.focus,.btn-danger:focus{color:#fff;background-color:#c82333;border-color:#bd2130;box-shadow:0 0 0 .2rem rgba(225,83,97,.5)}.btn-danger.disabled,.btn-danger:disabled{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:not(:disabled):not(.disabled).active,.btn-danger:not(:disabled):not(.disabled):active,.show>.btn-danger.dropdown-toggle{color:#fff;background-color:#bd2130;border-color:#b21f2d}.btn-danger:not(:disabled):not(.disabled).active:focus,.btn-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(225,83,97,.5)}.btn-light{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:hover{color:#212529;background-color:#e2e6ea;border-color:#dae0e5}.btn-light.focus,.btn-light:focus{color:#212529;background-color:#e2e6ea;border-color:#dae0e5;box-shadow:0 0 0 .2rem rgba(216,217,219,.5)}.btn-light.disabled,.btn-light:disabled{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:not(:disabled):not(.disabled).active,.btn-light:not(:disabled):not(.disabled):active,.show>.btn-light.dropdown-toggle{color:#212529;background-color:#dae0e5;border-color:#d3d9df}.btn-light:not(:disabled):not(.disabled).active:focus,.btn-light:not(:disabled):not(.disabled):active:focus,.show>.btn-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(216,217,219,.5)}.btn-dark{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:hover{color:#fff;background-color:#23272b;border-color:#1d2124}.btn-dark.focus,.btn-dark:focus{color:#fff;background-color:#23272b;border-color:#1d2124;box-shadow:0 0 0 .2rem rgba(82,88,93,.5)}.btn-dark.disabled,.btn-dark:disabled{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:not(:disabled):not(.disabled).active,.btn-dark:not(:disabled):not(.disabled):active,.show>.btn-dark.dropdown-toggle{color:#fff;background-color:#1d2124;border-color:#171a1d}.btn-dark:not(:disabled):not(.disabled).active:focus,.btn-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(82,88,93,.5)}.btn-outline-primary{color:#007bff;border-color:#007bff}.btn-outline-primary:hover{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary.focus,.btn-outline-primary:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-primary.disabled,.btn-outline-primary:disabled{color:#007bff;background-color:transparent}.btn-outline-primary:not(:disabled):not(.disabled).active,.btn-outline-primary:not(:disabled):not(.disabled):active,.show>.btn-outline-primary.dropdown-toggle{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary:not(:disabled):not(.disabled).active:focus,.btn-outline-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-secondary{color:#6c757d;border-color:#6c757d}.btn-outline-secondary:hover{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary.focus,.btn-outline-secondary:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-secondary.disabled,.btn-outline-secondary:disabled{color:#6c757d;background-color:transparent}.btn-outline-secondary:not(:disabled):not(.disabled).active,.btn-outline-secondary:not(:disabled):not(.disabled):active,.show>.btn-outline-secondary.dropdown-toggle{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary:not(:disabled):not(.disabled).active:focus,.btn-outline-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-success{color:#28a745;border-color:#28a745}.btn-outline-success:hover{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success.focus,.btn-outline-success:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-success.disabled,.btn-outline-success:disabled{color:#28a745;background-color:transparent}.btn-outline-success:not(:disabled):not(.disabled).active,.btn-outline-success:not(:disabled):not(.disabled):active,.show>.btn-outline-success.dropdown-toggle{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success:not(:disabled):not(.disabled).active:focus,.btn-outline-success:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-info{color:#17a2b8;border-color:#17a2b8}.btn-outline-info:hover{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info.focus,.btn-outline-info:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-info.disabled,.btn-outline-info:disabled{color:#17a2b8;background-color:transparent}.btn-outline-info:not(:disabled):not(.disabled).active,.btn-outline-info:not(:disabled):not(.disabled):active,.show>.btn-outline-info.dropdown-toggle{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info:not(:disabled):not(.disabled).active:focus,.btn-outline-info:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-warning{color:#ffc107;border-color:#ffc107}.btn-outline-warning:hover{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning.focus,.btn-outline-warning:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-warning.disabled,.btn-outline-warning:disabled{color:#ffc107;background-color:transparent}.btn-outline-warning:not(:disabled):not(.disabled).active,.btn-outline-warning:not(:disabled):not(.disabled):active,.show>.btn-outline-warning.dropdown-toggle{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning:not(:disabled):not(.disabled).active:focus,.btn-outline-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-danger{color:#dc3545;border-color:#dc3545}.btn-outline-danger:hover{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger.focus,.btn-outline-danger:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-danger.disabled,.btn-outline-danger:disabled{color:#dc3545;background-color:transparent}.btn-outline-danger:not(:disabled):not(.disabled).active,.btn-outline-danger:not(:disabled):not(.disabled):active,.show>.btn-outline-danger.dropdown-toggle{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger:not(:disabled):not(.disabled).active:focus,.btn-outline-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-light{color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:hover{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light.focus,.btn-outline-light:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-light.disabled,.btn-outline-light:disabled{color:#f8f9fa;background-color:transparent}.btn-outline-light:not(:disabled):not(.disabled).active,.btn-outline-light:not(:disabled):not(.disabled):active,.show>.btn-outline-light.dropdown-toggle{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:not(:disabled):not(.disabled).active:focus,.btn-outline-light:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-dark{color:#343a40;border-color:#343a40}.btn-outline-dark:hover{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark.focus,.btn-outline-dark:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-outline-dark.disabled,.btn-outline-dark:disabled{color:#343a40;background-color:transparent}.btn-outline-dark:not(:disabled):not(.disabled).active,.btn-outline-dark:not(:disabled):not(.disabled):active,.show>.btn-outline-dark.dropdown-toggle{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark:not(:disabled):not(.disabled).active:focus,.btn-outline-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-link{font-weight:400;color:#007bff;text-decoration:none}.btn-link:hover{color:#0056b3;text-decoration:underline}.btn-link.focus,.btn-link:focus{text-decoration:underline;box-shadow:none}.btn-link.disabled,.btn-link:disabled{color:#6c757d;pointer-events:none}.btn-group-lg>.btn,.btn-lg{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.btn-group-sm>.btn,.btn-sm{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:.5rem}input[type=button].btn-block,input[type=reset].btn-block,input[type=submit].btn-block{width:100%}.fade{transition:opacity .15s linear}@media (prefers-reduced-motion:reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{position:relative;height:0;overflow:hidden;transition:height .35s ease}@media (prefers-reduced-motion:reduce){.collapsing{transition:none}}.dropdown,.dropleft,.dropright,.dropup{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:10rem;padding:.5rem 0;margin:.125rem 0 0;font-size:1rem;color:#212529;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.15);border-radius:.25rem}.dropdown-menu-left{right:auto;left:0}.dropdown-menu-right{right:0;left:auto}@media (min-width:576px){.dropdown-menu-sm-left{right:auto;left:0}.dropdown-menu-sm-right{right:0;left:auto}}@media (min-width:768px){.dropdown-menu-md-left{right:auto;left:0}.dropdown-menu-md-right{right:0;left:auto}}@media (min-width:992px){.dropdown-menu-lg-left{right:auto;left:0}.dropdown-menu-lg-right{right:0;left:auto}}@media (min-width:1200px){.dropdown-menu-xl-left{right:auto;left:0}.dropdown-menu-xl-right{right:0;left:auto}}.dropup .dropdown-menu{top:auto;bottom:100%;margin-top:0;margin-bottom:.125rem}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-menu{top:0;right:auto;left:100%;margin-top:0;margin-left:.125rem}.dropright .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropright .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-toggle::after{vertical-align:0}.dropleft .dropdown-menu{top:0;right:100%;left:auto;margin-top:0;margin-right:.125rem}.dropleft .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropleft .dropdown-toggle::after{display:none}.dropleft .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropleft .dropdown-toggle:empty::after{margin-left:0}.dropleft .dropdown-toggle::before{vertical-align:0}.dropdown-menu[x-placement^=bottom],.dropdown-menu[x-placement^=left],.dropdown-menu[x-placement^=right],.dropdown-menu[x-placement^=top]{right:auto;bottom:auto}.dropdown-divider{height:0;margin:.5rem 0;overflow:hidden;border-top:1px solid #e9ecef}.dropdown-item{display:block;width:100%;padding:.25rem 1.5rem;clear:both;font-weight:400;color:#212529;text-align:inherit;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:focus,.dropdown-item:hover{color:#16181b;text-decoration:none;background-color:#f8f9fa}.dropdown-item.active,.dropdown-item:active{color:#fff;text-decoration:none;background-color:#007bff}.dropdown-item.disabled,.dropdown-item:disabled{color:#6c757d;pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:.5rem 1.5rem;margin-bottom:0;font-size:.875rem;color:#6c757d;white-space:nowrap}.dropdown-item-text{display:block;padding:.25rem 1.5rem;color:#212529}.btn-group,.btn-group-vertical{position:relative;display:-ms-inline-flexbox;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;-ms-flex:1 1 auto;flex:1 1 auto}.btn-group-vertical>.btn:hover,.btn-group>.btn:hover{z-index:1}.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus{z-index:1}.btn-toolbar{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-pack:start;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group>.btn-group:not(:first-child),.btn-group>.btn:not(:first-child){margin-left:-1px}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropright .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropleft .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:center;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn-group:not(:first-child),.btn-group-vertical>.btn:not(:first-child){margin-top:-1px}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn:not(:first-child){border-top-left-radius:0;border-top-right-radius:0}.btn-group-toggle>.btn,.btn-group-toggle>.btn-group>.btn{margin-bottom:0}.btn-group-toggle>.btn input[type=checkbox],.btn-group-toggle>.btn input[type=radio],.btn-group-toggle>.btn-group>.btn input[type=checkbox],.btn-group-toggle>.btn-group>.btn input[type=radio]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:stretch;align-items:stretch;width:100%}.input-group>.custom-file,.input-group>.custom-select,.input-group>.form-control,.input-group>.form-control-plaintext{position:relative;-ms-flex:1 1 0%;flex:1 1 0%;min-width:0;margin-bottom:0}.input-group>.custom-file+.custom-file,.input-group>.custom-file+.custom-select,.input-group>.custom-file+.form-control,.input-group>.custom-select+.custom-file,.input-group>.custom-select+.custom-select,.input-group>.custom-select+.form-control,.input-group>.form-control+.custom-file,.input-group>.form-control+.custom-select,.input-group>.form-control+.form-control,.input-group>.form-control-plaintext+.custom-file,.input-group>.form-control-plaintext+.custom-select,.input-group>.form-control-plaintext+.form-control{margin-left:-1px}.input-group>.custom-file .custom-file-input:focus~.custom-file-label,.input-group>.custom-select:focus,.input-group>.form-control:focus{z-index:3}.input-group>.custom-file .custom-file-input:focus{z-index:4}.input-group>.custom-select:not(:last-child),.input-group>.form-control:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-select:not(:first-child),.input-group>.form-control:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.custom-file{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center}.input-group>.custom-file:not(:last-child) .custom-file-label,.input-group>.custom-file:not(:last-child) .custom-file-label::after{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-file:not(:first-child) .custom-file-label{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-append,.input-group-prepend{display:-ms-flexbox;display:flex}.input-group-append .btn,.input-group-prepend .btn{position:relative;z-index:2}.input-group-append .btn:focus,.input-group-prepend .btn:focus{z-index:3}.input-group-append .btn+.btn,.input-group-append .btn+.input-group-text,.input-group-append .input-group-text+.btn,.input-group-append .input-group-text+.input-group-text,.input-group-prepend .btn+.btn,.input-group-prepend .btn+.input-group-text,.input-group-prepend .input-group-text+.btn,.input-group-prepend .input-group-text+.input-group-text{margin-left:-1px}.input-group-prepend{margin-right:-1px}.input-group-append{margin-left:-1px}.input-group-text{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.375rem .75rem;margin-bottom:0;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;text-align:center;white-space:nowrap;background-color:#e9ecef;border:1px solid #ced4da;border-radius:.25rem}.input-group-text input[type=checkbox],.input-group-text input[type=radio]{margin-top:0}.input-group-lg>.custom-select,.input-group-lg>.form-control:not(textarea){height:calc(1.5em + 1rem + 2px)}.input-group-lg>.custom-select,.input-group-lg>.form-control,.input-group-lg>.input-group-append>.btn,.input-group-lg>.input-group-append>.input-group-text,.input-group-lg>.input-group-prepend>.btn,.input-group-lg>.input-group-prepend>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.input-group-sm>.custom-select,.input-group-sm>.form-control:not(textarea){height:calc(1.5em + .5rem + 2px)}.input-group-sm>.custom-select,.input-group-sm>.form-control,.input-group-sm>.input-group-append>.btn,.input-group-sm>.input-group-append>.input-group-text,.input-group-sm>.input-group-prepend>.btn,.input-group-sm>.input-group-prepend>.input-group-text{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.input-group-lg>.custom-select,.input-group-sm>.custom-select{padding-right:1.75rem}.input-group>.input-group-append:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group>.input-group-append:last-child>.input-group-text:not(:last-child),.input-group>.input-group-append:not(:last-child)>.btn,.input-group>.input-group-append:not(:last-child)>.input-group-text,.input-group>.input-group-prepend>.btn,.input-group>.input-group-prepend>.input-group-text{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.input-group-append>.btn,.input-group>.input-group-append>.input-group-text,.input-group>.input-group-prepend:first-child>.btn:not(:first-child),.input-group>.input-group-prepend:first-child>.input-group-text:not(:first-child),.input-group>.input-group-prepend:not(:first-child)>.btn,.input-group>.input-group-prepend:not(:first-child)>.input-group-text{border-top-left-radius:0;border-bottom-left-radius:0}.custom-control{position:relative;display:block;min-height:1.5rem;padding-left:1.5rem}.custom-control-inline{display:-ms-inline-flexbox;display:inline-flex;margin-right:1rem}.custom-control-input{position:absolute;left:0;z-index:-1;width:1rem;height:1.25rem;opacity:0}.custom-control-input:checked~.custom-control-label::before{color:#fff;border-color:#007bff;background-color:#007bff}.custom-control-input:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-control-input:focus:not(:checked)~.custom-control-label::before{border-color:#80bdff}.custom-control-input:not(:disabled):active~.custom-control-label::before{color:#fff;background-color:#b3d7ff;border-color:#b3d7ff}.custom-control-input:disabled~.custom-control-label,.custom-control-input[disabled]~.custom-control-label{color:#6c757d}.custom-control-input:disabled~.custom-control-label::before,.custom-control-input[disabled]~.custom-control-label::before{background-color:#e9ecef}.custom-control-label{position:relative;margin-bottom:0;vertical-align:top}.custom-control-label::before{position:absolute;top:.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;pointer-events:none;content:"";background-color:#fff;border:#adb5bd solid 1px}.custom-control-label::after{position:absolute;top:.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;content:"";background:no-repeat 50%/50% 50%}.custom-checkbox .custom-control-label::before{border-radius:.25rem}.custom-checkbox .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%23fff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26l2.974 2.99L8 2.193z'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::before{border-color:#007bff;background-color:#007bff}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='4' viewBox='0 0 4 4'%3e%3cpath stroke='%23fff' d='M0 2h4'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-checkbox .custom-control-input:disabled:indeterminate~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-radio .custom-control-label::before{border-radius:50%}.custom-radio .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e")}.custom-radio .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-switch{padding-left:2.25rem}.custom-switch .custom-control-label::before{left:-2.25rem;width:1.75rem;pointer-events:all;border-radius:.5rem}.custom-switch .custom-control-label::after{top:calc(.25rem + 2px);left:calc(-2.25rem + 2px);width:calc(1rem - 4px);height:calc(1rem - 4px);background-color:#adb5bd;border-radius:.5rem;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-transform .15s ease-in-out;transition:transform .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:transform .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-transform .15s ease-in-out}@media (prefers-reduced-motion:reduce){.custom-switch .custom-control-label::after{transition:none}}.custom-switch .custom-control-input:checked~.custom-control-label::after{background-color:#fff;-webkit-transform:translateX(.75rem);transform:translateX(.75rem)}.custom-switch .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-select{display:inline-block;width:100%;height:calc(1.5em + .75rem + 2px);padding:.375rem 1.75rem .375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;vertical-align:middle;background:#fff url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px;border:1px solid #ced4da;border-radius:.25rem;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-select:focus{border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-select:focus::-ms-value{color:#495057;background-color:#fff}.custom-select[multiple],.custom-select[size]:not([size="1"]){height:auto;padding-right:.75rem;background-image:none}.custom-select:disabled{color:#6c757d;background-color:#e9ecef}.custom-select::-ms-expand{display:none}.custom-select:-moz-focusring{color:transparent;text-shadow:0 0 0 #495057}.custom-select-sm{height:calc(1.5em + .5rem + 2px);padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem}.custom-select-lg{height:calc(1.5em + 1rem + 2px);padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem}.custom-file{position:relative;display:inline-block;width:100%;height:calc(1.5em + .75rem + 2px);margin-bottom:0}.custom-file-input{position:relative;z-index:2;width:100%;height:calc(1.5em + .75rem + 2px);margin:0;opacity:0}.custom-file-input:focus~.custom-file-label{border-color:#80bdff;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-file-input:disabled~.custom-file-label,.custom-file-input[disabled]~.custom-file-label{background-color:#e9ecef}.custom-file-input:lang(en)~.custom-file-label::after{content:"Browse"}.custom-file-input~.custom-file-label[data-browse]::after{content:attr(data-browse)}.custom-file-label{position:absolute;top:0;right:0;left:0;z-index:1;height:calc(1.5em + .75rem + 2px);padding:.375rem .75rem;font-weight:400;line-height:1.5;color:#495057;background-color:#fff;border:1px solid #ced4da;border-radius:.25rem}.custom-file-label::after{position:absolute;top:0;right:0;bottom:0;z-index:3;display:block;height:calc(1.5em + .75rem);padding:.375rem .75rem;line-height:1.5;color:#495057;content:"Browse";background-color:#e9ecef;border-left:inherit;border-radius:0 .25rem .25rem 0}.custom-range{width:100%;height:1.4rem;padding:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-range:focus{outline:0}.custom-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range:focus::-ms-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range::-moz-focus-outer{border:0}.custom-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;background-color:#007bff;border:0;border-radius:1rem;-webkit-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;-webkit-appearance:none;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-webkit-slider-thumb{-webkit-transition:none;transition:none}}.custom-range::-webkit-slider-thumb:active{background-color:#b3d7ff}.custom-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-moz-range-thumb{width:1rem;height:1rem;background-color:#007bff;border:0;border-radius:1rem;-moz-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;-moz-appearance:none;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-moz-range-thumb{-moz-transition:none;transition:none}}.custom-range::-moz-range-thumb:active{background-color:#b3d7ff}.custom-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-ms-thumb{width:1rem;height:1rem;margin-top:0;margin-right:.2rem;margin-left:.2rem;background-color:#007bff;border:0;border-radius:1rem;-ms-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-ms-thumb{-ms-transition:none;transition:none}}.custom-range::-ms-thumb:active{background-color:#b3d7ff}.custom-range::-ms-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:transparent;border-color:transparent;border-width:.5rem}.custom-range::-ms-fill-lower{background-color:#dee2e6;border-radius:1rem}.custom-range::-ms-fill-upper{margin-right:15px;background-color:#dee2e6;border-radius:1rem}.custom-range:disabled::-webkit-slider-thumb{background-color:#adb5bd}.custom-range:disabled::-webkit-slider-runnable-track{cursor:default}.custom-range:disabled::-moz-range-thumb{background-color:#adb5bd}.custom-range:disabled::-moz-range-track{cursor:default}.custom-range:disabled::-ms-thumb{background-color:#adb5bd}.custom-control-label::before,.custom-file-label,.custom-select{transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.custom-control-label::before,.custom-file-label,.custom-select{transition:none}}.nav{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:.5rem 1rem}.nav-link:focus,.nav-link:hover{text-decoration:none}.nav-link.disabled{color:#6c757d;pointer-events:none;cursor:default}.nav-tabs{border-bottom:1px solid #dee2e6}.nav-tabs .nav-item{margin-bottom:-1px}.nav-tabs .nav-link{border:1px solid transparent;border-top-left-radius:.25rem;border-top-right-radius:.25rem}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{border-color:#e9ecef #e9ecef #dee2e6}.nav-tabs .nav-link.disabled{color:#6c757d;background-color:transparent;border-color:transparent}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:#495057;background-color:#fff;border-color:#dee2e6 #dee2e6 #fff}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.nav-pills .nav-link{border-radius:.25rem}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:#fff;background-color:#007bff}.nav-fill .nav-item{-ms-flex:1 1 auto;flex:1 1 auto;text-align:center}.nav-justified .nav-item{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;text-align:center}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between;padding:.5rem 1rem}.navbar .container,.navbar .container-fluid,.navbar .container-lg,.navbar .container-md,.navbar .container-sm,.navbar .container-xl{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between}.navbar-brand{display:inline-block;padding-top:.3125rem;padding-bottom:.3125rem;margin-right:1rem;font-size:1.25rem;line-height:inherit;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{text-decoration:none}.navbar-nav{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link{padding-right:0;padding-left:0}.navbar-nav .dropdown-menu{position:static;float:none}.navbar-text{display:inline-block;padding-top:.5rem;padding-bottom:.5rem}.navbar-collapse{-ms-flex-preferred-size:100%;flex-basis:100%;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center}.navbar-toggler{padding:.25rem .75rem;font-size:1.25rem;line-height:1;background-color:transparent;border:1px solid transparent;border-radius:.25rem}.navbar-toggler:focus,.navbar-toggler:hover{text-decoration:none}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;content:"";background:no-repeat center center;background-size:100% 100%}@media (max-width:575.98px){.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid,.navbar-expand-sm>.container-lg,.navbar-expand-sm>.container-md,.navbar-expand-sm>.container-sm,.navbar-expand-sm>.container-xl{padding-right:0;padding-left:0}}@media (min-width:576px){.navbar-expand-sm{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-sm .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid,.navbar-expand-sm>.container-lg,.navbar-expand-sm>.container-md,.navbar-expand-sm>.container-sm,.navbar-expand-sm>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-sm .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}}@media (max-width:767.98px){.navbar-expand-md>.container,.navbar-expand-md>.container-fluid,.navbar-expand-md>.container-lg,.navbar-expand-md>.container-md,.navbar-expand-md>.container-sm,.navbar-expand-md>.container-xl{padding-right:0;padding-left:0}}@media (min-width:768px){.navbar-expand-md{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-md .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-md>.container,.navbar-expand-md>.container-fluid,.navbar-expand-md>.container-lg,.navbar-expand-md>.container-md,.navbar-expand-md>.container-sm,.navbar-expand-md>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-md .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}}@media (max-width:991.98px){.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid,.navbar-expand-lg>.container-lg,.navbar-expand-lg>.container-md,.navbar-expand-lg>.container-sm,.navbar-expand-lg>.container-xl{padding-right:0;padding-left:0}}@media (min-width:992px){.navbar-expand-lg{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-lg .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid,.navbar-expand-lg>.container-lg,.navbar-expand-lg>.container-md,.navbar-expand-lg>.container-sm,.navbar-expand-lg>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-lg .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}}@media (max-width:1199.98px){.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid,.navbar-expand-xl>.container-lg,.navbar-expand-xl>.container-md,.navbar-expand-xl>.container-sm,.navbar-expand-xl>.container-xl{padding-right:0;padding-left:0}}@media (min-width:1200px){.navbar-expand-xl{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-xl .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid,.navbar-expand-xl>.container-lg,.navbar-expand-xl>.container-md,.navbar-expand-xl>.container-sm,.navbar-expand-xl>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-xl .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}}.navbar-expand{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand>.container,.navbar-expand>.container-fluid,.navbar-expand>.container-lg,.navbar-expand>.container-md,.navbar-expand>.container-sm,.navbar-expand>.container-xl{padding-right:0;padding-left:0}.navbar-expand .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand>.container,.navbar-expand>.container-fluid,.navbar-expand>.container-lg,.navbar-expand>.container-md,.navbar-expand>.container-sm,.navbar-expand>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-light .navbar-brand{color:rgba(0,0,0,.9)}.navbar-light .navbar-brand:focus,.navbar-light .navbar-brand:hover{color:rgba(0,0,0,.9)}.navbar-light .navbar-nav .nav-link{color:rgba(0,0,0,.5)}.navbar-light .navbar-nav .nav-link:focus,.navbar-light .navbar-nav .nav-link:hover{color:rgba(0,0,0,.7)}.navbar-light .navbar-nav .nav-link.disabled{color:rgba(0,0,0,.3)}.navbar-light .navbar-nav .active>.nav-link,.navbar-light .navbar-nav .nav-link.active,.navbar-light .navbar-nav .nav-link.show,.navbar-light .navbar-nav .show>.nav-link{color:rgba(0,0,0,.9)}.navbar-light .navbar-toggler{color:rgba(0,0,0,.5);border-color:rgba(0,0,0,.1)}.navbar-light .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='30' height='30' viewBox='0 0 30 30'%3e%3cpath stroke='rgba(0, 0, 0, 0.5)' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-light .navbar-text{color:rgba(0,0,0,.5)}.navbar-light .navbar-text a{color:rgba(0,0,0,.9)}.navbar-light .navbar-text a:focus,.navbar-light .navbar-text a:hover{color:rgba(0,0,0,.9)}.navbar-dark .navbar-brand{color:#fff}.navbar-dark .navbar-brand:focus,.navbar-dark .navbar-brand:hover{color:#fff}.navbar-dark .navbar-nav .nav-link{color:rgba(255,255,255,.5)}.navbar-dark .navbar-nav .nav-link:focus,.navbar-dark .navbar-nav .nav-link:hover{color:rgba(255,255,255,.75)}.navbar-dark .navbar-nav .nav-link.disabled{color:rgba(255,255,255,.25)}.navbar-dark .navbar-nav .active>.nav-link,.navbar-dark .navbar-nav .nav-link.active,.navbar-dark .navbar-nav .nav-link.show,.navbar-dark .navbar-nav .show>.nav-link{color:#fff}.navbar-dark .navbar-toggler{color:rgba(255,255,255,.5);border-color:rgba(255,255,255,.1)}.navbar-dark .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='30' height='30' viewBox='0 0 30 30'%3e%3cpath stroke='rgba(255, 255, 255, 0.5)' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-dark .navbar-text{color:rgba(255,255,255,.5)}.navbar-dark .navbar-text a{color:#fff}.navbar-dark .navbar-text a:focus,.navbar-dark .navbar-text a:hover{color:#fff}.card{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;min-width:0;word-wrap:break-word;background-color:#fff;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}.card>hr{margin-right:0;margin-left:0}.card>.list-group:first-child .list-group-item:first-child{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.card>.list-group:last-child .list-group-item:last-child{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.card-body{-ms-flex:1 1 auto;flex:1 1 auto;min-height:1px;padding:1.25rem}.card-title{margin-bottom:.75rem}.card-subtitle{margin-top:-.375rem;margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link:hover{text-decoration:none}.card-link+.card-link{margin-left:1.25rem}.card-header{padding:.75rem 1.25rem;margin-bottom:0;background-color:rgba(0,0,0,.03);border-bottom:1px solid rgba(0,0,0,.125)}.card-header:first-child{border-radius:calc(.25rem - 1px) calc(.25rem - 1px) 0 0}.card-header+.list-group .list-group-item:first-child{border-top:0}.card-footer{padding:.75rem 1.25rem;background-color:rgba(0,0,0,.03);border-top:1px solid rgba(0,0,0,.125)}.card-footer:last-child{border-radius:0 0 calc(.25rem - 1px) calc(.25rem - 1px)}.card-header-tabs{margin-right:-.625rem;margin-bottom:-.75rem;margin-left:-.625rem;border-bottom:0}.card-header-pills{margin-right:-.625rem;margin-left:-.625rem}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:1.25rem}.card-img,.card-img-bottom,.card-img-top{-ms-flex-negative:0;flex-shrink:0;width:100%}.card-img,.card-img-top{border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card-img,.card-img-bottom{border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card-deck .card{margin-bottom:15px}@media (min-width:576px){.card-deck{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap;margin-right:-15px;margin-left:-15px}.card-deck .card{-ms-flex:1 0 0%;flex:1 0 0%;margin-right:15px;margin-bottom:0;margin-left:15px}}.card-group>.card{margin-bottom:15px}@media (min-width:576px){.card-group{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap}.card-group>.card{-ms-flex:1 0 0%;flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:not(:last-child) .card-header,.card-group>.card:not(:last-child) .card-img-top{border-top-right-radius:0}.card-group>.card:not(:last-child) .card-footer,.card-group>.card:not(:last-child) .card-img-bottom{border-bottom-right-radius:0}.card-group>.card:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:not(:first-child) .card-header,.card-group>.card:not(:first-child) .card-img-top{border-top-left-radius:0}.card-group>.card:not(:first-child) .card-footer,.card-group>.card:not(:first-child) .card-img-bottom{border-bottom-left-radius:0}}.card-columns .card{margin-bottom:.75rem}@media (min-width:576px){.card-columns{-webkit-column-count:3;-moz-column-count:3;column-count:3;-webkit-column-gap:1.25rem;-moz-column-gap:1.25rem;column-gap:1.25rem;orphans:1;widows:1}.card-columns .card{display:inline-block;width:100%}}.accordion>.card{overflow:hidden}.accordion>.card:not(:last-of-type){border-bottom:0;border-bottom-right-radius:0;border-bottom-left-radius:0}.accordion>.card:not(:first-of-type){border-top-left-radius:0;border-top-right-radius:0}.accordion>.card>.card-header{border-radius:0;margin-bottom:-1px}.breadcrumb{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding:.75rem 1rem;margin-bottom:1rem;list-style:none;background-color:#e9ecef;border-radius:.25rem}.breadcrumb-item+.breadcrumb-item{padding-left:.5rem}.breadcrumb-item+.breadcrumb-item::before{display:inline-block;padding-right:.5rem;color:#6c757d;content:"/"}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:underline}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:none}.breadcrumb-item.active{color:#6c757d}.pagination{display:-ms-flexbox;display:flex;padding-left:0;list-style:none;border-radius:.25rem}.page-link{position:relative;display:block;padding:.5rem .75rem;margin-left:-1px;line-height:1.25;color:#007bff;background-color:#fff;border:1px solid #dee2e6}.page-link:hover{z-index:2;color:#0056b3;text-decoration:none;background-color:#e9ecef;border-color:#dee2e6}.page-link:focus{z-index:3;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.page-item:first-child .page-link{margin-left:0;border-top-left-radius:.25rem;border-bottom-left-radius:.25rem}.page-item:last-child .page-link{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.page-item.active .page-link{z-index:3;color:#fff;background-color:#007bff;border-color:#007bff}.page-item.disabled .page-link{color:#6c757d;pointer-events:none;cursor:auto;background-color:#fff;border-color:#dee2e6}.pagination-lg .page-link{padding:.75rem 1.5rem;font-size:1.25rem;line-height:1.5}.pagination-lg .page-item:first-child .page-link{border-top-left-radius:.3rem;border-bottom-left-radius:.3rem}.pagination-lg .page-item:last-child .page-link{border-top-right-radius:.3rem;border-bottom-right-radius:.3rem}.pagination-sm .page-link{padding:.25rem .5rem;font-size:.875rem;line-height:1.5}.pagination-sm .page-item:first-child .page-link{border-top-left-radius:.2rem;border-bottom-left-radius:.2rem}.pagination-sm .page-item:last-child .page-link{border-top-right-radius:.2rem;border-bottom-right-radius:.2rem}.badge{display:inline-block;padding:.25em .4em;font-size:75%;font-weight:700;line-height:1;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.badge{transition:none}}a.badge:focus,a.badge:hover{text-decoration:none}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.badge-pill{padding-right:.6em;padding-left:.6em;border-radius:10rem}.badge-primary{color:#fff;background-color:#007bff}a.badge-primary:focus,a.badge-primary:hover{color:#fff;background-color:#0062cc}a.badge-primary.focus,a.badge-primary:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.badge-secondary{color:#fff;background-color:#6c757d}a.badge-secondary:focus,a.badge-secondary:hover{color:#fff;background-color:#545b62}a.badge-secondary.focus,a.badge-secondary:focus{outline:0;box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.badge-success{color:#fff;background-color:#28a745}a.badge-success:focus,a.badge-success:hover{color:#fff;background-color:#1e7e34}a.badge-success.focus,a.badge-success:focus{outline:0;box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.badge-info{color:#fff;background-color:#17a2b8}a.badge-info:focus,a.badge-info:hover{color:#fff;background-color:#117a8b}a.badge-info.focus,a.badge-info:focus{outline:0;box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.badge-warning{color:#212529;background-color:#ffc107}a.badge-warning:focus,a.badge-warning:hover{color:#212529;background-color:#d39e00}a.badge-warning.focus,a.badge-warning:focus{outline:0;box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.badge-danger{color:#fff;background-color:#dc3545}a.badge-danger:focus,a.badge-danger:hover{color:#fff;background-color:#bd2130}a.badge-danger.focus,a.badge-danger:focus{outline:0;box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.badge-light{color:#212529;background-color:#f8f9fa}a.badge-light:focus,a.badge-light:hover{color:#212529;background-color:#dae0e5}a.badge-light.focus,a.badge-light:focus{outline:0;box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.badge-dark{color:#fff;background-color:#343a40}a.badge-dark:focus,a.badge-dark:hover{color:#fff;background-color:#1d2124}a.badge-dark.focus,a.badge-dark:focus{outline:0;box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.jumbotron{padding:2rem 1rem;margin-bottom:2rem;background-color:#e9ecef;border-radius:.3rem}@media (min-width:576px){.jumbotron{padding:4rem 2rem}}.jumbotron-fluid{padding-right:0;padding-left:0;border-radius:0}.alert{position:relative;padding:.75rem 1.25rem;margin-bottom:1rem;border:1px solid transparent;border-radius:.25rem}.alert-heading{color:inherit}.alert-link{font-weight:700}.alert-dismissible{padding-right:4rem}.alert-dismissible .close{position:absolute;top:0;right:0;padding:.75rem 1.25rem;color:inherit}.alert-primary{color:#004085;background-color:#cce5ff;border-color:#b8daff}.alert-primary hr{border-top-color:#9fcdff}.alert-primary .alert-link{color:#002752}.alert-secondary{color:#383d41;background-color:#e2e3e5;border-color:#d6d8db}.alert-secondary hr{border-top-color:#c8cbcf}.alert-secondary .alert-link{color:#202326}.alert-success{color:#155724;background-color:#d4edda;border-color:#c3e6cb}.alert-success hr{border-top-color:#b1dfbb}.alert-success .alert-link{color:#0b2e13}.alert-info{color:#0c5460;background-color:#d1ecf1;border-color:#bee5eb}.alert-info hr{border-top-color:#abdde5}.alert-info .alert-link{color:#062c33}.alert-warning{color:#856404;background-color:#fff3cd;border-color:#ffeeba}.alert-warning hr{border-top-color:#ffe8a1}.alert-warning .alert-link{color:#533f03}.alert-danger{color:#721c24;background-color:#f8d7da;border-color:#f5c6cb}.alert-danger hr{border-top-color:#f1b0b7}.alert-danger .alert-link{color:#491217}.alert-light{color:#818182;background-color:#fefefe;border-color:#fdfdfe}.alert-light hr{border-top-color:#ececf6}.alert-light .alert-link{color:#686868}.alert-dark{color:#1b1e21;background-color:#d6d8d9;border-color:#c6c8ca}.alert-dark hr{border-top-color:#b9bbbe}.alert-dark .alert-link{color:#040505}@-webkit-keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}.progress{display:-ms-flexbox;display:flex;height:1rem;overflow:hidden;font-size:.75rem;background-color:#e9ecef;border-radius:.25rem}.progress-bar{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;overflow:hidden;color:#fff;text-align:center;white-space:nowrap;background-color:#007bff;transition:width .6s ease}@media (prefers-reduced-motion:reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:1rem 1rem}.progress-bar-animated{-webkit-animation:progress-bar-stripes 1s linear infinite;animation:progress-bar-stripes 1s linear infinite}@media (prefers-reduced-motion:reduce){.progress-bar-animated{-webkit-animation:none;animation:none}}.media{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start}.media-body{-ms-flex:1;flex:1}.list-group{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0}.list-group-item-action{width:100%;color:#495057;text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{z-index:1;color:#495057;text-decoration:none;background-color:#f8f9fa}.list-group-item-action:active{color:#212529;background-color:#e9ecef}.list-group-item{position:relative;display:block;padding:.75rem 1.25rem;background-color:#fff;border:1px solid rgba(0,0,0,.125)}.list-group-item:first-child{border-top-left-radius:.25rem;border-top-right-radius:.25rem}.list-group-item:last-child{border-bottom-right-radius:.25rem;border-bottom-left-radius:.25rem}.list-group-item.disabled,.list-group-item:disabled{color:#6c757d;pointer-events:none;background-color:#fff}.list-group-item.active{z-index:2;color:#fff;background-color:#007bff;border-color:#007bff}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:-1px;border-top-width:1px}.list-group-horizontal{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal .list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal .list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal .list-group-item.active{margin-top:0}.list-group-horizontal .list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal .list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}@media (min-width:576px){.list-group-horizontal-sm{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-sm .list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-sm .list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-sm .list-group-item.active{margin-top:0}.list-group-horizontal-sm .list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-sm .list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:768px){.list-group-horizontal-md{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-md .list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-md .list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-md .list-group-item.active{margin-top:0}.list-group-horizontal-md .list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-md .list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:992px){.list-group-horizontal-lg{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-lg .list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-lg .list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-lg .list-group-item.active{margin-top:0}.list-group-horizontal-lg .list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-lg .list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:1200px){.list-group-horizontal-xl{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-xl .list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-xl .list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-xl .list-group-item.active{margin-top:0}.list-group-horizontal-xl .list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-xl .list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}.list-group-flush .list-group-item{border-right-width:0;border-left-width:0;border-radius:0}.list-group-flush .list-group-item:first-child{border-top-width:0}.list-group-flush:last-child .list-group-item:last-child{border-bottom-width:0}.list-group-item-primary{color:#004085;background-color:#b8daff}.list-group-item-primary.list-group-item-action:focus,.list-group-item-primary.list-group-item-action:hover{color:#004085;background-color:#9fcdff}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#004085;border-color:#004085}.list-group-item-secondary{color:#383d41;background-color:#d6d8db}.list-group-item-secondary.list-group-item-action:focus,.list-group-item-secondary.list-group-item-action:hover{color:#383d41;background-color:#c8cbcf}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#383d41;border-color:#383d41}.list-group-item-success{color:#155724;background-color:#c3e6cb}.list-group-item-success.list-group-item-action:focus,.list-group-item-success.list-group-item-action:hover{color:#155724;background-color:#b1dfbb}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#155724;border-color:#155724}.list-group-item-info{color:#0c5460;background-color:#bee5eb}.list-group-item-info.list-group-item-action:focus,.list-group-item-info.list-group-item-action:hover{color:#0c5460;background-color:#abdde5}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#0c5460;border-color:#0c5460}.list-group-item-warning{color:#856404;background-color:#ffeeba}.list-group-item-warning.list-group-item-action:focus,.list-group-item-warning.list-group-item-action:hover{color:#856404;background-color:#ffe8a1}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#856404;border-color:#856404}.list-group-item-danger{color:#721c24;background-color:#f5c6cb}.list-group-item-danger.list-group-item-action:focus,.list-group-item-danger.list-group-item-action:hover{color:#721c24;background-color:#f1b0b7}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#721c24;border-color:#721c24}.list-group-item-light{color:#818182;background-color:#fdfdfe}.list-group-item-light.list-group-item-action:focus,.list-group-item-light.list-group-item-action:hover{color:#818182;background-color:#ececf6}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#818182;border-color:#818182}.list-group-item-dark{color:#1b1e21;background-color:#c6c8ca}.list-group-item-dark.list-group-item-action:focus,.list-group-item-dark.list-group-item-action:hover{color:#1b1e21;background-color:#b9bbbe}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#1b1e21;border-color:#1b1e21}.close{float:right;font-size:1.5rem;font-weight:700;line-height:1;color:#000;text-shadow:0 1px 0 #fff;opacity:.5}.close:hover{color:#000;text-decoration:none}.close:not(:disabled):not(.disabled):focus,.close:not(:disabled):not(.disabled):hover{opacity:.75}button.close{padding:0;background-color:transparent;border:0;-webkit-appearance:none;-moz-appearance:none;appearance:none}a.close.disabled{pointer-events:none}.toast{max-width:350px;overflow:hidden;font-size:.875rem;background-color:rgba(255,255,255,.85);background-clip:padding-box;border:1px solid rgba(0,0,0,.1);box-shadow:0 .25rem .75rem rgba(0,0,0,.1);-webkit-backdrop-filter:blur(10px);backdrop-filter:blur(10px);opacity:0;border-radius:.25rem}.toast:not(:last-child){margin-bottom:.75rem}.toast.showing{opacity:1}.toast.show{display:block;opacity:1}.toast.hide{display:none}.toast-header{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.25rem .75rem;color:#6c757d;background-color:rgba(255,255,255,.85);background-clip:padding-box;border-bottom:1px solid rgba(0,0,0,.05)}.toast-body{padding:.75rem}.modal-open{overflow:hidden}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal{position:fixed;top:0;left:0;z-index:1050;display:none;width:100%;height:100%;overflow:hidden;outline:0}.modal-dialog{position:relative;width:auto;margin:.5rem;pointer-events:none}.modal.fade .modal-dialog{transition:-webkit-transform .3s ease-out;transition:transform .3s ease-out;transition:transform .3s ease-out,-webkit-transform .3s ease-out;-webkit-transform:translate(0,-50px);transform:translate(0,-50px)}@media (prefers-reduced-motion:reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{-webkit-transform:none;transform:none}.modal.modal-static .modal-dialog{-webkit-transform:scale(1.02);transform:scale(1.02)}.modal-dialog-scrollable{display:-ms-flexbox;display:flex;max-height:calc(100% - 1rem)}.modal-dialog-scrollable .modal-content{max-height:calc(100vh - 1rem);overflow:hidden}.modal-dialog-scrollable .modal-footer,.modal-dialog-scrollable .modal-header{-ms-flex-negative:0;flex-shrink:0}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;min-height:calc(100% - 1rem)}.modal-dialog-centered::before{display:block;height:calc(100vh - 1rem);content:""}.modal-dialog-centered.modal-dialog-scrollable{-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;height:100%}.modal-dialog-centered.modal-dialog-scrollable .modal-content{max-height:none}.modal-dialog-centered.modal-dialog-scrollable::before{content:none}.modal-content{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;width:100%;pointer-events:auto;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem;outline:0}.modal-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:.5}.modal-header{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:justify;justify-content:space-between;padding:1rem 1rem;border-bottom:1px solid #dee2e6;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.modal-header .close{padding:1rem 1rem;margin:-1rem -1rem -1rem auto}.modal-title{margin-bottom:0;line-height:1.5}.modal-body{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;padding:1rem}.modal-footer{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:end;justify-content:flex-end;padding:.75rem;border-top:1px solid #dee2e6;border-bottom-right-radius:calc(.3rem - 1px);border-bottom-left-radius:calc(.3rem - 1px)}.modal-footer>*{margin:.25rem}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width:576px){.modal-dialog{max-width:500px;margin:1.75rem auto}.modal-dialog-scrollable{max-height:calc(100% - 3.5rem)}.modal-dialog-scrollable .modal-content{max-height:calc(100vh - 3.5rem)}.modal-dialog-centered{min-height:calc(100% - 3.5rem)}.modal-dialog-centered::before{height:calc(100vh - 3.5rem)}.modal-sm{max-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{max-width:800px}}@media (min-width:1200px){.modal-xl{max-width:1140px}}.tooltip{position:absolute;z-index:1070;display:block;margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;opacity:0}.tooltip.show{opacity:.9}.tooltip .arrow{position:absolute;display:block;width:.8rem;height:.4rem}.tooltip .arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[x-placement^=top],.bs-tooltip-top{padding:.4rem 0}.bs-tooltip-auto[x-placement^=top] .arrow,.bs-tooltip-top .arrow{bottom:0}.bs-tooltip-auto[x-placement^=top] .arrow::before,.bs-tooltip-top .arrow::before{top:0;border-width:.4rem .4rem 0;border-top-color:#000}.bs-tooltip-auto[x-placement^=right],.bs-tooltip-right{padding:0 .4rem}.bs-tooltip-auto[x-placement^=right] .arrow,.bs-tooltip-right .arrow{left:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=right] .arrow::before,.bs-tooltip-right .arrow::before{right:0;border-width:.4rem .4rem .4rem 0;border-right-color:#000}.bs-tooltip-auto[x-placement^=bottom],.bs-tooltip-bottom{padding:.4rem 0}.bs-tooltip-auto[x-placement^=bottom] .arrow,.bs-tooltip-bottom .arrow{top:0}.bs-tooltip-auto[x-placement^=bottom] .arrow::before,.bs-tooltip-bottom .arrow::before{bottom:0;border-width:0 .4rem .4rem;border-bottom-color:#000}.bs-tooltip-auto[x-placement^=left],.bs-tooltip-left{padding:0 .4rem}.bs-tooltip-auto[x-placement^=left] .arrow,.bs-tooltip-left .arrow{right:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=left] .arrow::before,.bs-tooltip-left .arrow::before{left:0;border-width:.4rem 0 .4rem .4rem;border-left-color:#000}.tooltip-inner{max-width:200px;padding:.25rem .5rem;color:#fff;text-align:center;background-color:#000;border-radius:.25rem}.popover{position:absolute;top:0;left:0;z-index:1060;display:block;max-width:276px;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem}.popover .arrow{position:absolute;display:block;width:1rem;height:.5rem;margin:0 .3rem}.popover .arrow::after,.popover .arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid}.bs-popover-auto[x-placement^=top],.bs-popover-top{margin-bottom:.5rem}.bs-popover-auto[x-placement^=top]>.arrow,.bs-popover-top>.arrow{bottom:calc(-.5rem - 1px)}.bs-popover-auto[x-placement^=top]>.arrow::before,.bs-popover-top>.arrow::before{bottom:0;border-width:.5rem .5rem 0;border-top-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=top]>.arrow::after,.bs-popover-top>.arrow::after{bottom:1px;border-width:.5rem .5rem 0;border-top-color:#fff}.bs-popover-auto[x-placement^=right],.bs-popover-right{margin-left:.5rem}.bs-popover-auto[x-placement^=right]>.arrow,.bs-popover-right>.arrow{left:calc(-.5rem - 1px);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=right]>.arrow::before,.bs-popover-right>.arrow::before{left:0;border-width:.5rem .5rem .5rem 0;border-right-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=right]>.arrow::after,.bs-popover-right>.arrow::after{left:1px;border-width:.5rem .5rem .5rem 0;border-right-color:#fff}.bs-popover-auto[x-placement^=bottom],.bs-popover-bottom{margin-top:.5rem}.bs-popover-auto[x-placement^=bottom]>.arrow,.bs-popover-bottom>.arrow{top:calc(-.5rem - 1px)}.bs-popover-auto[x-placement^=bottom]>.arrow::before,.bs-popover-bottom>.arrow::before{top:0;border-width:0 .5rem .5rem .5rem;border-bottom-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=bottom]>.arrow::after,.bs-popover-bottom>.arrow::after{top:1px;border-width:0 .5rem .5rem .5rem;border-bottom-color:#fff}.bs-popover-auto[x-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:1rem;margin-left:-.5rem;content:"";border-bottom:1px solid #f7f7f7}.bs-popover-auto[x-placement^=left],.bs-popover-left{margin-right:.5rem}.bs-popover-auto[x-placement^=left]>.arrow,.bs-popover-left>.arrow{right:calc(-.5rem - 1px);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=left]>.arrow::before,.bs-popover-left>.arrow::before{right:0;border-width:.5rem 0 .5rem .5rem;border-left-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=left]>.arrow::after,.bs-popover-left>.arrow::after{right:1px;border-width:.5rem 0 .5rem .5rem;border-left-color:#fff}.popover-header{padding:.5rem .75rem;margin-bottom:0;font-size:1rem;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.popover-header:empty{display:none}.popover-body{padding:.5rem .75rem;color:#212529}.carousel{position:relative}.carousel.pointer-event{-ms-touch-action:pan-y;touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden;transition:-webkit-transform .6s ease-in-out;transition:transform .6s ease-in-out;transition:transform .6s ease-in-out,-webkit-transform .6s ease-in-out}@media (prefers-reduced-motion:reduce){.carousel-item{transition:none}}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.active.carousel-item-right,.carousel-item-next:not(.carousel-item-left){-webkit-transform:translateX(100%);transform:translateX(100%)}.active.carousel-item-left,.carousel-item-prev:not(.carousel-item-right){-webkit-transform:translateX(-100%);transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;-webkit-transform:none;transform:none}.carousel-fade .carousel-item-next.carousel-item-left,.carousel-fade .carousel-item-prev.carousel-item-right,.carousel-fade .carousel-item.active{z-index:1;opacity:1}.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{z-index:0;opacity:0;transition:opacity 0s .6s}@media (prefers-reduced-motion:reduce){.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{transition:none}}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;z-index:1;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:15%;color:#fff;text-align:center;opacity:.5;transition:opacity .15s ease}@media (prefers-reduced-motion:reduce){.carousel-control-next,.carousel-control-prev{transition:none}}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:20px;height:20px;background:no-repeat 50%/100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath d='M5.25 0l-4 4 4 4 1.5-1.5L4.25 4l2.5-2.5L5.25 0z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath d='M2.75 0l-1.5 1.5L3.75 4l-2.5 2.5L2.75 8l4-4-4-4z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:15;display:-ms-flexbox;display:flex;-ms-flex-pack:center;justify-content:center;padding-left:0;margin-right:15%;margin-left:15%;list-style:none}.carousel-indicators li{box-sizing:content-box;-ms-flex:0 1 auto;flex:0 1 auto;width:30px;height:3px;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5;transition:opacity .6s ease}@media (prefers-reduced-motion:reduce){.carousel-indicators li{transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center}@-webkit-keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}.spinner-border{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;border:.25em solid currentColor;border-right-color:transparent;border-radius:50%;-webkit-animation:spinner-border .75s linear infinite;animation:spinner-border .75s linear infinite}.spinner-border-sm{width:1rem;height:1rem;border-width:.2em}@-webkit-keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1}}@keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1}}.spinner-grow{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;background-color:currentColor;border-radius:50%;opacity:0;-webkit-animation:spinner-grow .75s linear infinite;animation:spinner-grow .75s linear infinite}.spinner-grow-sm{width:1rem;height:1rem}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.bg-primary{background-color:#007bff!important}a.bg-primary:focus,a.bg-primary:hover,button.bg-primary:focus,button.bg-primary:hover{background-color:#0062cc!important}.bg-secondary{background-color:#6c757d!important}a.bg-secondary:focus,a.bg-secondary:hover,button.bg-secondary:focus,button.bg-secondary:hover{background-color:#545b62!important}.bg-success{background-color:#28a745!important}a.bg-success:focus,a.bg-success:hover,button.bg-success:focus,button.bg-success:hover{background-color:#1e7e34!important}.bg-info{background-color:#17a2b8!important}a.bg-info:focus,a.bg-info:hover,button.bg-info:focus,button.bg-info:hover{background-color:#117a8b!important}.bg-warning{background-color:#ffc107!important}a.bg-warning:focus,a.bg-warning:hover,button.bg-warning:focus,button.bg-warning:hover{background-color:#d39e00!important}.bg-danger{background-color:#dc3545!important}a.bg-danger:focus,a.bg-danger:hover,button.bg-danger:focus,button.bg-danger:hover{background-color:#bd2130!important}.bg-light{background-color:#f8f9fa!important}a.bg-light:focus,a.bg-light:hover,button.bg-light:focus,button.bg-light:hover{background-color:#dae0e5!important}.bg-dark{background-color:#343a40!important}a.bg-dark:focus,a.bg-dark:hover,button.bg-dark:focus,button.bg-dark:hover{background-color:#1d2124!important}.bg-white{background-color:#fff!important}.bg-transparent{background-color:transparent!important}.border{border:1px solid #dee2e6!important}.border-top{border-top:1px solid #dee2e6!important}.border-right{border-right:1px solid #dee2e6!important}.border-bottom{border-bottom:1px solid #dee2e6!important}.border-left{border-left:1px solid #dee2e6!important}.border-0{border:0!important}.border-top-0{border-top:0!important}.border-right-0{border-right:0!important}.border-bottom-0{border-bottom:0!important}.border-left-0{border-left:0!important}.border-primary{border-color:#007bff!important}.border-secondary{border-color:#6c757d!important}.border-success{border-color:#28a745!important}.border-info{border-color:#17a2b8!important}.border-warning{border-color:#ffc107!important}.border-danger{border-color:#dc3545!important}.border-light{border-color:#f8f9fa!important}.border-dark{border-color:#343a40!important}.border-white{border-color:#fff!important}.rounded-sm{border-radius:.2rem!important}.rounded{border-radius:.25rem!important}.rounded-top{border-top-left-radius:.25rem!important;border-top-right-radius:.25rem!important}.rounded-right{border-top-right-radius:.25rem!important;border-bottom-right-radius:.25rem!important}.rounded-bottom{border-bottom-right-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-left{border-top-left-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-lg{border-radius:.3rem!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:50rem!important}.rounded-0{border-radius:0!important}.clearfix::after{display:block;clear:both;content:""}.d-none{display:none!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:-ms-flexbox!important;display:flex!important}.d-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}@media (min-width:576px){.d-sm-none{display:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:-ms-flexbox!important;display:flex!important}.d-sm-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:768px){.d-md-none{display:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:-ms-flexbox!important;display:flex!important}.d-md-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:992px){.d-lg-none{display:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:-ms-flexbox!important;display:flex!important}.d-lg-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:1200px){.d-xl-none{display:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:-ms-flexbox!important;display:flex!important}.d-xl-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media print{.d-print-none{display:none!important}.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:-ms-flexbox!important;display:flex!important}.d-print-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}.embed-responsive{position:relative;display:block;width:100%;padding:0;overflow:hidden}.embed-responsive::before{display:block;content:""}.embed-responsive .embed-responsive-item,.embed-responsive embed,.embed-responsive iframe,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-21by9::before{padding-top:42.857143%}.embed-responsive-16by9::before{padding-top:56.25%}.embed-responsive-4by3::before{padding-top:75%}.embed-responsive-1by1::before{padding-top:100%}.flex-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-center{-ms-flex-align:center!important;align-items:center!important}.align-items-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}@media (min-width:576px){.flex-sm-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-sm-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-sm-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-sm-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-sm-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-sm-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-sm-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-sm-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-sm-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-sm-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-sm-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-sm-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-sm-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-sm-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-sm-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-sm-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-sm-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-sm-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-sm-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-sm-center{-ms-flex-align:center!important;align-items:center!important}.align-items-sm-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-sm-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-sm-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-sm-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-sm-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-sm-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-sm-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-sm-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-sm-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-sm-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-sm-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-sm-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-sm-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-sm-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:768px){.flex-md-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-md-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-md-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-md-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-md-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-md-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-md-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-md-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-md-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-md-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-md-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-md-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-md-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-md-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-md-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-md-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-md-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-md-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-md-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-md-center{-ms-flex-align:center!important;align-items:center!important}.align-items-md-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-md-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-md-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-md-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-md-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-md-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-md-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-md-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-md-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-md-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-md-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-md-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-md-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-md-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:992px){.flex-lg-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-lg-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-lg-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-lg-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-lg-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-lg-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-lg-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-lg-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-lg-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-lg-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-lg-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-lg-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-lg-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-lg-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-lg-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-lg-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-lg-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-lg-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-lg-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-lg-center{-ms-flex-align:center!important;align-items:center!important}.align-items-lg-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-lg-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-lg-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-lg-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-lg-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-lg-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-lg-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-lg-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-lg-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-lg-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-lg-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-lg-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-lg-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-lg-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:1200px){.flex-xl-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-xl-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-xl-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-xl-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-xl-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-xl-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-xl-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-xl-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-xl-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-xl-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-xl-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-xl-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-xl-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-xl-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-xl-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-xl-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-xl-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-xl-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-xl-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-xl-center{-ms-flex-align:center!important;align-items:center!important}.align-items-xl-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-xl-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-xl-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-xl-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-xl-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-xl-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-xl-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-xl-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-xl-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-xl-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-xl-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-xl-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-xl-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-xl-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}.float-left{float:left!important}.float-right{float:right!important}.float-none{float:none!important}@media (min-width:576px){.float-sm-left{float:left!important}.float-sm-right{float:right!important}.float-sm-none{float:none!important}}@media (min-width:768px){.float-md-left{float:left!important}.float-md-right{float:right!important}.float-md-none{float:none!important}}@media (min-width:992px){.float-lg-left{float:left!important}.float-lg-right{float:right!important}.float-lg-none{float:none!important}}@media (min-width:1200px){.float-xl-left{float:left!important}.float-xl-right{float:right!important}.float-xl-none{float:none!important}}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}@supports ((position:-webkit-sticky) or (position:sticky)){.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;overflow:visible;clip:auto;white-space:normal}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075)!important}.shadow{box-shadow:0 .5rem 1rem rgba(0,0,0,.15)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175)!important}.shadow-none{box-shadow:none!important}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mw-100{max-width:100%!important}.mh-100{max-height:100%!important}.min-vw-100{min-width:100vw!important}.min-vh-100{min-height:100vh!important}.vw-100{width:100vw!important}.vh-100{height:100vh!important}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;pointer-events:auto;content:"";background-color:rgba(0,0,0,0)}.m-0{margin:0!important}.mt-0,.my-0{margin-top:0!important}.mr-0,.mx-0{margin-right:0!important}.mb-0,.my-0{margin-bottom:0!important}.ml-0,.mx-0{margin-left:0!important}.m-1{margin:.25rem!important}.mt-1,.my-1{margin-top:.25rem!important}.mr-1,.mx-1{margin-right:.25rem!important}.mb-1,.my-1{margin-bottom:.25rem!important}.ml-1,.mx-1{margin-left:.25rem!important}.m-2{margin:.5rem!important}.mt-2,.my-2{margin-top:.5rem!important}.mr-2,.mx-2{margin-right:.5rem!important}.mb-2,.my-2{margin-bottom:.5rem!important}.ml-2,.mx-2{margin-left:.5rem!important}.m-3{margin:1rem!important}.mt-3,.my-3{margin-top:1rem!important}.mr-3,.mx-3{margin-right:1rem!important}.mb-3,.my-3{margin-bottom:1rem!important}.ml-3,.mx-3{margin-left:1rem!important}.m-4{margin:1.5rem!important}.mt-4,.my-4{margin-top:1.5rem!important}.mr-4,.mx-4{margin-right:1.5rem!important}.mb-4,.my-4{margin-bottom:1.5rem!important}.ml-4,.mx-4{margin-left:1.5rem!important}.m-5{margin:3rem!important}.mt-5,.my-5{margin-top:3rem!important}.mr-5,.mx-5{margin-right:3rem!important}.mb-5,.my-5{margin-bottom:3rem!important}.ml-5,.mx-5{margin-left:3rem!important}.p-0{padding:0!important}.pt-0,.py-0{padding-top:0!important}.pr-0,.px-0{padding-right:0!important}.pb-0,.py-0{padding-bottom:0!important}.pl-0,.px-0{padding-left:0!important}.p-1{padding:.25rem!important}.pt-1,.py-1{padding-top:.25rem!important}.pr-1,.px-1{padding-right:.25rem!important}.pb-1,.py-1{padding-bottom:.25rem!important}.pl-1,.px-1{padding-left:.25rem!important}.p-2{padding:.5rem!important}.pt-2,.py-2{padding-top:.5rem!important}.pr-2,.px-2{padding-right:.5rem!important}.pb-2,.py-2{padding-bottom:.5rem!important}.pl-2,.px-2{padding-left:.5rem!important}.p-3{padding:1rem!important}.pt-3,.py-3{padding-top:1rem!important}.pr-3,.px-3{padding-right:1rem!important}.pb-3,.py-3{padding-bottom:1rem!important}.pl-3,.px-3{padding-left:1rem!important}.p-4{padding:1.5rem!important}.pt-4,.py-4{padding-top:1.5rem!important}.pr-4,.px-4{padding-right:1.5rem!important}.pb-4,.py-4{padding-bottom:1.5rem!important}.pl-4,.px-4{padding-left:1.5rem!important}.p-5{padding:3rem!important}.pt-5,.py-5{padding-top:3rem!important}.pr-5,.px-5{padding-right:3rem!important}.pb-5,.py-5{padding-bottom:3rem!important}.pl-5,.px-5{padding-left:3rem!important}.m-n1{margin:-.25rem!important}.mt-n1,.my-n1{margin-top:-.25rem!important}.mr-n1,.mx-n1{margin-right:-.25rem!important}.mb-n1,.my-n1{margin-bottom:-.25rem!important}.ml-n1,.mx-n1{margin-left:-.25rem!important}.m-n2{margin:-.5rem!important}.mt-n2,.my-n2{margin-top:-.5rem!important}.mr-n2,.mx-n2{margin-right:-.5rem!important}.mb-n2,.my-n2{margin-bottom:-.5rem!important}.ml-n2,.mx-n2{margin-left:-.5rem!important}.m-n3{margin:-1rem!important}.mt-n3,.my-n3{margin-top:-1rem!important}.mr-n3,.mx-n3{margin-right:-1rem!important}.mb-n3,.my-n3{margin-bottom:-1rem!important}.ml-n3,.mx-n3{margin-left:-1rem!important}.m-n4{margin:-1.5rem!important}.mt-n4,.my-n4{margin-top:-1.5rem!important}.mr-n4,.mx-n4{margin-right:-1.5rem!important}.mb-n4,.my-n4{margin-bottom:-1.5rem!important}.ml-n4,.mx-n4{margin-left:-1.5rem!important}.m-n5{margin:-3rem!important}.mt-n5,.my-n5{margin-top:-3rem!important}.mr-n5,.mx-n5{margin-right:-3rem!important}.mb-n5,.my-n5{margin-bottom:-3rem!important}.ml-n5,.mx-n5{margin-left:-3rem!important}.m-auto{margin:auto!important}.mt-auto,.my-auto{margin-top:auto!important}.mr-auto,.mx-auto{margin-right:auto!important}.mb-auto,.my-auto{margin-bottom:auto!important}.ml-auto,.mx-auto{margin-left:auto!important}@media (min-width:576px){.m-sm-0{margin:0!important}.mt-sm-0,.my-sm-0{margin-top:0!important}.mr-sm-0,.mx-sm-0{margin-right:0!important}.mb-sm-0,.my-sm-0{margin-bottom:0!important}.ml-sm-0,.mx-sm-0{margin-left:0!important}.m-sm-1{margin:.25rem!important}.mt-sm-1,.my-sm-1{margin-top:.25rem!important}.mr-sm-1,.mx-sm-1{margin-right:.25rem!important}.mb-sm-1,.my-sm-1{margin-bottom:.25rem!important}.ml-sm-1,.mx-sm-1{margin-left:.25rem!important}.m-sm-2{margin:.5rem!important}.mt-sm-2,.my-sm-2{margin-top:.5rem!important}.mr-sm-2,.mx-sm-2{margin-right:.5rem!important}.mb-sm-2,.my-sm-2{margin-bottom:.5rem!important}.ml-sm-2,.mx-sm-2{margin-left:.5rem!important}.m-sm-3{margin:1rem!important}.mt-sm-3,.my-sm-3{margin-top:1rem!important}.mr-sm-3,.mx-sm-3{margin-right:1rem!important}.mb-sm-3,.my-sm-3{margin-bottom:1rem!important}.ml-sm-3,.mx-sm-3{margin-left:1rem!important}.m-sm-4{margin:1.5rem!important}.mt-sm-4,.my-sm-4{margin-top:1.5rem!important}.mr-sm-4,.mx-sm-4{margin-right:1.5rem!important}.mb-sm-4,.my-sm-4{margin-bottom:1.5rem!important}.ml-sm-4,.mx-sm-4{margin-left:1.5rem!important}.m-sm-5{margin:3rem!important}.mt-sm-5,.my-sm-5{margin-top:3rem!important}.mr-sm-5,.mx-sm-5{margin-right:3rem!important}.mb-sm-5,.my-sm-5{margin-bottom:3rem!important}.ml-sm-5,.mx-sm-5{margin-left:3rem!important}.p-sm-0{padding:0!important}.pt-sm-0,.py-sm-0{padding-top:0!important}.pr-sm-0,.px-sm-0{padding-right:0!important}.pb-sm-0,.py-sm-0{padding-bottom:0!important}.pl-sm-0,.px-sm-0{padding-left:0!important}.p-sm-1{padding:.25rem!important}.pt-sm-1,.py-sm-1{padding-top:.25rem!important}.pr-sm-1,.px-sm-1{padding-right:.25rem!important}.pb-sm-1,.py-sm-1{padding-bottom:.25rem!important}.pl-sm-1,.px-sm-1{padding-left:.25rem!important}.p-sm-2{padding:.5rem!important}.pt-sm-2,.py-sm-2{padding-top:.5rem!important}.pr-sm-2,.px-sm-2{padding-right:.5rem!important}.pb-sm-2,.py-sm-2{padding-bottom:.5rem!important}.pl-sm-2,.px-sm-2{padding-left:.5rem!important}.p-sm-3{padding:1rem!important}.pt-sm-3,.py-sm-3{padding-top:1rem!important}.pr-sm-3,.px-sm-3{padding-right:1rem!important}.pb-sm-3,.py-sm-3{padding-bottom:1rem!important}.pl-sm-3,.px-sm-3{padding-left:1rem!important}.p-sm-4{padding:1.5rem!important}.pt-sm-4,.py-sm-4{padding-top:1.5rem!important}.pr-sm-4,.px-sm-4{padding-right:1.5rem!important}.pb-sm-4,.py-sm-4{padding-bottom:1.5rem!important}.pl-sm-4,.px-sm-4{padding-left:1.5rem!important}.p-sm-5{padding:3rem!important}.pt-sm-5,.py-sm-5{padding-top:3rem!important}.pr-sm-5,.px-sm-5{padding-right:3rem!important}.pb-sm-5,.py-sm-5{padding-bottom:3rem!important}.pl-sm-5,.px-sm-5{padding-left:3rem!important}.m-sm-n1{margin:-.25rem!important}.mt-sm-n1,.my-sm-n1{margin-top:-.25rem!important}.mr-sm-n1,.mx-sm-n1{margin-right:-.25rem!important}.mb-sm-n1,.my-sm-n1{margin-bottom:-.25rem!important}.ml-sm-n1,.mx-sm-n1{margin-left:-.25rem!important}.m-sm-n2{margin:-.5rem!important}.mt-sm-n2,.my-sm-n2{margin-top:-.5rem!important}.mr-sm-n2,.mx-sm-n2{margin-right:-.5rem!important}.mb-sm-n2,.my-sm-n2{margin-bottom:-.5rem!important}.ml-sm-n2,.mx-sm-n2{margin-left:-.5rem!important}.m-sm-n3{margin:-1rem!important}.mt-sm-n3,.my-sm-n3{margin-top:-1rem!important}.mr-sm-n3,.mx-sm-n3{margin-right:-1rem!important}.mb-sm-n3,.my-sm-n3{margin-bottom:-1rem!important}.ml-sm-n3,.mx-sm-n3{margin-left:-1rem!important}.m-sm-n4{margin:-1.5rem!important}.mt-sm-n4,.my-sm-n4{margin-top:-1.5rem!important}.mr-sm-n4,.mx-sm-n4{margin-right:-1.5rem!important}.mb-sm-n4,.my-sm-n4{margin-bottom:-1.5rem!important}.ml-sm-n4,.mx-sm-n4{margin-left:-1.5rem!important}.m-sm-n5{margin:-3rem!important}.mt-sm-n5,.my-sm-n5{margin-top:-3rem!important}.mr-sm-n5,.mx-sm-n5{margin-right:-3rem!important}.mb-sm-n5,.my-sm-n5{margin-bottom:-3rem!important}.ml-sm-n5,.mx-sm-n5{margin-left:-3rem!important}.m-sm-auto{margin:auto!important}.mt-sm-auto,.my-sm-auto{margin-top:auto!important}.mr-sm-auto,.mx-sm-auto{margin-right:auto!important}.mb-sm-auto,.my-sm-auto{margin-bottom:auto!important}.ml-sm-auto,.mx-sm-auto{margin-left:auto!important}}@media (min-width:768px){.m-md-0{margin:0!important}.mt-md-0,.my-md-0{margin-top:0!important}.mr-md-0,.mx-md-0{margin-right:0!important}.mb-md-0,.my-md-0{margin-bottom:0!important}.ml-md-0,.mx-md-0{margin-left:0!important}.m-md-1{margin:.25rem!important}.mt-md-1,.my-md-1{margin-top:.25rem!important}.mr-md-1,.mx-md-1{margin-right:.25rem!important}.mb-md-1,.my-md-1{margin-bottom:.25rem!important}.ml-md-1,.mx-md-1{margin-left:.25rem!important}.m-md-2{margin:.5rem!important}.mt-md-2,.my-md-2{margin-top:.5rem!important}.mr-md-2,.mx-md-2{margin-right:.5rem!important}.mb-md-2,.my-md-2{margin-bottom:.5rem!important}.ml-md-2,.mx-md-2{margin-left:.5rem!important}.m-md-3{margin:1rem!important}.mt-md-3,.my-md-3{margin-top:1rem!important}.mr-md-3,.mx-md-3{margin-right:1rem!important}.mb-md-3,.my-md-3{margin-bottom:1rem!important}.ml-md-3,.mx-md-3{margin-left:1rem!important}.m-md-4{margin:1.5rem!important}.mt-md-4,.my-md-4{margin-top:1.5rem!important}.mr-md-4,.mx-md-4{margin-right:1.5rem!important}.mb-md-4,.my-md-4{margin-bottom:1.5rem!important}.ml-md-4,.mx-md-4{margin-left:1.5rem!important}.m-md-5{margin:3rem!important}.mt-md-5,.my-md-5{margin-top:3rem!important}.mr-md-5,.mx-md-5{margin-right:3rem!important}.mb-md-5,.my-md-5{margin-bottom:3rem!important}.ml-md-5,.mx-md-5{margin-left:3rem!important}.p-md-0{padding:0!important}.pt-md-0,.py-md-0{padding-top:0!important}.pr-md-0,.px-md-0{padding-right:0!important}.pb-md-0,.py-md-0{padding-bottom:0!important}.pl-md-0,.px-md-0{padding-left:0!important}.p-md-1{padding:.25rem!important}.pt-md-1,.py-md-1{padding-top:.25rem!important}.pr-md-1,.px-md-1{padding-right:.25rem!important}.pb-md-1,.py-md-1{padding-bottom:.25rem!important}.pl-md-1,.px-md-1{padding-left:.25rem!important}.p-md-2{padding:.5rem!important}.pt-md-2,.py-md-2{padding-top:.5rem!important}.pr-md-2,.px-md-2{padding-right:.5rem!important}.pb-md-2,.py-md-2{padding-bottom:.5rem!important}.pl-md-2,.px-md-2{padding-left:.5rem!important}.p-md-3{padding:1rem!important}.pt-md-3,.py-md-3{padding-top:1rem!important}.pr-md-3,.px-md-3{padding-right:1rem!important}.pb-md-3,.py-md-3{padding-bottom:1rem!important}.pl-md-3,.px-md-3{padding-left:1rem!important}.p-md-4{padding:1.5rem!important}.pt-md-4,.py-md-4{padding-top:1.5rem!important}.pr-md-4,.px-md-4{padding-right:1.5rem!important}.pb-md-4,.py-md-4{padding-bottom:1.5rem!important}.pl-md-4,.px-md-4{padding-left:1.5rem!important}.p-md-5{padding:3rem!important}.pt-md-5,.py-md-5{padding-top:3rem!important}.pr-md-5,.px-md-5{padding-right:3rem!important}.pb-md-5,.py-md-5{padding-bottom:3rem!important}.pl-md-5,.px-md-5{padding-left:3rem!important}.m-md-n1{margin:-.25rem!important}.mt-md-n1,.my-md-n1{margin-top:-.25rem!important}.mr-md-n1,.mx-md-n1{margin-right:-.25rem!important}.mb-md-n1,.my-md-n1{margin-bottom:-.25rem!important}.ml-md-n1,.mx-md-n1{margin-left:-.25rem!important}.m-md-n2{margin:-.5rem!important}.mt-md-n2,.my-md-n2{margin-top:-.5rem!important}.mr-md-n2,.mx-md-n2{margin-right:-.5rem!important}.mb-md-n2,.my-md-n2{margin-bottom:-.5rem!important}.ml-md-n2,.mx-md-n2{margin-left:-.5rem!important}.m-md-n3{margin:-1rem!important}.mt-md-n3,.my-md-n3{margin-top:-1rem!important}.mr-md-n3,.mx-md-n3{margin-right:-1rem!important}.mb-md-n3,.my-md-n3{margin-bottom:-1rem!important}.ml-md-n3,.mx-md-n3{margin-left:-1rem!important}.m-md-n4{margin:-1.5rem!important}.mt-md-n4,.my-md-n4{margin-top:-1.5rem!important}.mr-md-n4,.mx-md-n4{margin-right:-1.5rem!important}.mb-md-n4,.my-md-n4{margin-bottom:-1.5rem!important}.ml-md-n4,.mx-md-n4{margin-left:-1.5rem!important}.m-md-n5{margin:-3rem!important}.mt-md-n5,.my-md-n5{margin-top:-3rem!important}.mr-md-n5,.mx-md-n5{margin-right:-3rem!important}.mb-md-n5,.my-md-n5{margin-bottom:-3rem!important}.ml-md-n5,.mx-md-n5{margin-left:-3rem!important}.m-md-auto{margin:auto!important}.mt-md-auto,.my-md-auto{margin-top:auto!important}.mr-md-auto,.mx-md-auto{margin-right:auto!important}.mb-md-auto,.my-md-auto{margin-bottom:auto!important}.ml-md-auto,.mx-md-auto{margin-left:auto!important}}@media (min-width:992px){.m-lg-0{margin:0!important}.mt-lg-0,.my-lg-0{margin-top:0!important}.mr-lg-0,.mx-lg-0{margin-right:0!important}.mb-lg-0,.my-lg-0{margin-bottom:0!important}.ml-lg-0,.mx-lg-0{margin-left:0!important}.m-lg-1{margin:.25rem!important}.mt-lg-1,.my-lg-1{margin-top:.25rem!important}.mr-lg-1,.mx-lg-1{margin-right:.25rem!important}.mb-lg-1,.my-lg-1{margin-bottom:.25rem!important}.ml-lg-1,.mx-lg-1{margin-left:.25rem!important}.m-lg-2{margin:.5rem!important}.mt-lg-2,.my-lg-2{margin-top:.5rem!important}.mr-lg-2,.mx-lg-2{margin-right:.5rem!important}.mb-lg-2,.my-lg-2{margin-bottom:.5rem!important}.ml-lg-2,.mx-lg-2{margin-left:.5rem!important}.m-lg-3{margin:1rem!important}.mt-lg-3,.my-lg-3{margin-top:1rem!important}.mr-lg-3,.mx-lg-3{margin-right:1rem!important}.mb-lg-3,.my-lg-3{margin-bottom:1rem!important}.ml-lg-3,.mx-lg-3{margin-left:1rem!important}.m-lg-4{margin:1.5rem!important}.mt-lg-4,.my-lg-4{margin-top:1.5rem!important}.mr-lg-4,.mx-lg-4{margin-right:1.5rem!important}.mb-lg-4,.my-lg-4{margin-bottom:1.5rem!important}.ml-lg-4,.mx-lg-4{margin-left:1.5rem!important}.m-lg-5{margin:3rem!important}.mt-lg-5,.my-lg-5{margin-top:3rem!important}.mr-lg-5,.mx-lg-5{margin-right:3rem!important}.mb-lg-5,.my-lg-5{margin-bottom:3rem!important}.ml-lg-5,.mx-lg-5{margin-left:3rem!important}.p-lg-0{padding:0!important}.pt-lg-0,.py-lg-0{padding-top:0!important}.pr-lg-0,.px-lg-0{padding-right:0!important}.pb-lg-0,.py-lg-0{padding-bottom:0!important}.pl-lg-0,.px-lg-0{padding-left:0!important}.p-lg-1{padding:.25rem!important}.pt-lg-1,.py-lg-1{padding-top:.25rem!important}.pr-lg-1,.px-lg-1{padding-right:.25rem!important}.pb-lg-1,.py-lg-1{padding-bottom:.25rem!important}.pl-lg-1,.px-lg-1{padding-left:.25rem!important}.p-lg-2{padding:.5rem!important}.pt-lg-2,.py-lg-2{padding-top:.5rem!important}.pr-lg-2,.px-lg-2{padding-right:.5rem!important}.pb-lg-2,.py-lg-2{padding-bottom:.5rem!important}.pl-lg-2,.px-lg-2{padding-left:.5rem!important}.p-lg-3{padding:1rem!important}.pt-lg-3,.py-lg-3{padding-top:1rem!important}.pr-lg-3,.px-lg-3{padding-right:1rem!important}.pb-lg-3,.py-lg-3{padding-bottom:1rem!important}.pl-lg-3,.px-lg-3{padding-left:1rem!important}.p-lg-4{padding:1.5rem!important}.pt-lg-4,.py-lg-4{padding-top:1.5rem!important}.pr-lg-4,.px-lg-4{padding-right:1.5rem!important}.pb-lg-4,.py-lg-4{padding-bottom:1.5rem!important}.pl-lg-4,.px-lg-4{padding-left:1.5rem!important}.p-lg-5{padding:3rem!important}.pt-lg-5,.py-lg-5{padding-top:3rem!important}.pr-lg-5,.px-lg-5{padding-right:3rem!important}.pb-lg-5,.py-lg-5{padding-bottom:3rem!important}.pl-lg-5,.px-lg-5{padding-left:3rem!important}.m-lg-n1{margin:-.25rem!important}.mt-lg-n1,.my-lg-n1{margin-top:-.25rem!important}.mr-lg-n1,.mx-lg-n1{margin-right:-.25rem!important}.mb-lg-n1,.my-lg-n1{margin-bottom:-.25rem!important}.ml-lg-n1,.mx-lg-n1{margin-left:-.25rem!important}.m-lg-n2{margin:-.5rem!important}.mt-lg-n2,.my-lg-n2{margin-top:-.5rem!important}.mr-lg-n2,.mx-lg-n2{margin-right:-.5rem!important}.mb-lg-n2,.my-lg-n2{margin-bottom:-.5rem!important}.ml-lg-n2,.mx-lg-n2{margin-left:-.5rem!important}.m-lg-n3{margin:-1rem!important}.mt-lg-n3,.my-lg-n3{margin-top:-1rem!important}.mr-lg-n3,.mx-lg-n3{margin-right:-1rem!important}.mb-lg-n3,.my-lg-n3{margin-bottom:-1rem!important}.ml-lg-n3,.mx-lg-n3{margin-left:-1rem!important}.m-lg-n4{margin:-1.5rem!important}.mt-lg-n4,.my-lg-n4{margin-top:-1.5rem!important}.mr-lg-n4,.mx-lg-n4{margin-right:-1.5rem!important}.mb-lg-n4,.my-lg-n4{margin-bottom:-1.5rem!important}.ml-lg-n4,.mx-lg-n4{margin-left:-1.5rem!important}.m-lg-n5{margin:-3rem!important}.mt-lg-n5,.my-lg-n5{margin-top:-3rem!important}.mr-lg-n5,.mx-lg-n5{margin-right:-3rem!important}.mb-lg-n5,.my-lg-n5{margin-bottom:-3rem!important}.ml-lg-n5,.mx-lg-n5{margin-left:-3rem!important}.m-lg-auto{margin:auto!important}.mt-lg-auto,.my-lg-auto{margin-top:auto!important}.mr-lg-auto,.mx-lg-auto{margin-right:auto!important}.mb-lg-auto,.my-lg-auto{margin-bottom:auto!important}.ml-lg-auto,.mx-lg-auto{margin-left:auto!important}}@media (min-width:1200px){.m-xl-0{margin:0!important}.mt-xl-0,.my-xl-0{margin-top:0!important}.mr-xl-0,.mx-xl-0{margin-right:0!important}.mb-xl-0,.my-xl-0{margin-bottom:0!important}.ml-xl-0,.mx-xl-0{margin-left:0!important}.m-xl-1{margin:.25rem!important}.mt-xl-1,.my-xl-1{margin-top:.25rem!important}.mr-xl-1,.mx-xl-1{margin-right:.25rem!important}.mb-xl-1,.my-xl-1{margin-bottom:.25rem!important}.ml-xl-1,.mx-xl-1{margin-left:.25rem!important}.m-xl-2{margin:.5rem!important}.mt-xl-2,.my-xl-2{margin-top:.5rem!important}.mr-xl-2,.mx-xl-2{margin-right:.5rem!important}.mb-xl-2,.my-xl-2{margin-bottom:.5rem!important}.ml-xl-2,.mx-xl-2{margin-left:.5rem!important}.m-xl-3{margin:1rem!important}.mt-xl-3,.my-xl-3{margin-top:1rem!important}.mr-xl-3,.mx-xl-3{margin-right:1rem!important}.mb-xl-3,.my-xl-3{margin-bottom:1rem!important}.ml-xl-3,.mx-xl-3{margin-left:1rem!important}.m-xl-4{margin:1.5rem!important}.mt-xl-4,.my-xl-4{margin-top:1.5rem!important}.mr-xl-4,.mx-xl-4{margin-right:1.5rem!important}.mb-xl-4,.my-xl-4{margin-bottom:1.5rem!important}.ml-xl-4,.mx-xl-4{margin-left:1.5rem!important}.m-xl-5{margin:3rem!important}.mt-xl-5,.my-xl-5{margin-top:3rem!important}.mr-xl-5,.mx-xl-5{margin-right:3rem!important}.mb-xl-5,.my-xl-5{margin-bottom:3rem!important}.ml-xl-5,.mx-xl-5{margin-left:3rem!important}.p-xl-0{padding:0!important}.pt-xl-0,.py-xl-0{padding-top:0!important}.pr-xl-0,.px-xl-0{padding-right:0!important}.pb-xl-0,.py-xl-0{padding-bottom:0!important}.pl-xl-0,.px-xl-0{padding-left:0!important}.p-xl-1{padding:.25rem!important}.pt-xl-1,.py-xl-1{padding-top:.25rem!important}.pr-xl-1,.px-xl-1{padding-right:.25rem!important}.pb-xl-1,.py-xl-1{padding-bottom:.25rem!important}.pl-xl-1,.px-xl-1{padding-left:.25rem!important}.p-xl-2{padding:.5rem!important}.pt-xl-2,.py-xl-2{padding-top:.5rem!important}.pr-xl-2,.px-xl-2{padding-right:.5rem!important}.pb-xl-2,.py-xl-2{padding-bottom:.5rem!important}.pl-xl-2,.px-xl-2{padding-left:.5rem!important}.p-xl-3{padding:1rem!important}.pt-xl-3,.py-xl-3{padding-top:1rem!important}.pr-xl-3,.px-xl-3{padding-right:1rem!important}.pb-xl-3,.py-xl-3{padding-bottom:1rem!important}.pl-xl-3,.px-xl-3{padding-left:1rem!important}.p-xl-4{padding:1.5rem!important}.pt-xl-4,.py-xl-4{padding-top:1.5rem!important}.pr-xl-4,.px-xl-4{padding-right:1.5rem!important}.pb-xl-4,.py-xl-4{padding-bottom:1.5rem!important}.pl-xl-4,.px-xl-4{padding-left:1.5rem!important}.p-xl-5{padding:3rem!important}.pt-xl-5,.py-xl-5{padding-top:3rem!important}.pr-xl-5,.px-xl-5{padding-right:3rem!important}.pb-xl-5,.py-xl-5{padding-bottom:3rem!important}.pl-xl-5,.px-xl-5{padding-left:3rem!important}.m-xl-n1{margin:-.25rem!important}.mt-xl-n1,.my-xl-n1{margin-top:-.25rem!important}.mr-xl-n1,.mx-xl-n1{margin-right:-.25rem!important}.mb-xl-n1,.my-xl-n1{margin-bottom:-.25rem!important}.ml-xl-n1,.mx-xl-n1{margin-left:-.25rem!important}.m-xl-n2{margin:-.5rem!important}.mt-xl-n2,.my-xl-n2{margin-top:-.5rem!important}.mr-xl-n2,.mx-xl-n2{margin-right:-.5rem!important}.mb-xl-n2,.my-xl-n2{margin-bottom:-.5rem!important}.ml-xl-n2,.mx-xl-n2{margin-left:-.5rem!important}.m-xl-n3{margin:-1rem!important}.mt-xl-n3,.my-xl-n3{margin-top:-1rem!important}.mr-xl-n3,.mx-xl-n3{margin-right:-1rem!important}.mb-xl-n3,.my-xl-n3{margin-bottom:-1rem!important}.ml-xl-n3,.mx-xl-n3{margin-left:-1rem!important}.m-xl-n4{margin:-1.5rem!important}.mt-xl-n4,.my-xl-n4{margin-top:-1.5rem!important}.mr-xl-n4,.mx-xl-n4{margin-right:-1.5rem!important}.mb-xl-n4,.my-xl-n4{margin-bottom:-1.5rem!important}.ml-xl-n4,.mx-xl-n4{margin-left:-1.5rem!important}.m-xl-n5{margin:-3rem!important}.mt-xl-n5,.my-xl-n5{margin-top:-3rem!important}.mr-xl-n5,.mx-xl-n5{margin-right:-3rem!important}.mb-xl-n5,.my-xl-n5{margin-bottom:-3rem!important}.ml-xl-n5,.mx-xl-n5{margin-left:-3rem!important}.m-xl-auto{margin:auto!important}.mt-xl-auto,.my-xl-auto{margin-top:auto!important}.mr-xl-auto,.mx-xl-auto{margin-right:auto!important}.mb-xl-auto,.my-xl-auto{margin-bottom:auto!important}.ml-xl-auto,.mx-xl-auto{margin-left:auto!important}}.text-monospace{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace!important}.text-justify{text-align:justify!important}.text-wrap{white-space:normal!important}.text-nowrap{white-space:nowrap!important}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.text-left{text-align:left!important}.text-right{text-align:right!important}.text-center{text-align:center!important}@media (min-width:576px){.text-sm-left{text-align:left!important}.text-sm-right{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.text-md-left{text-align:left!important}.text-md-right{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.text-lg-left{text-align:left!important}.text-lg-right{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.text-xl-left{text-align:left!important}.text-xl-right{text-align:right!important}.text-xl-center{text-align:center!important}}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.font-weight-light{font-weight:300!important}.font-weight-lighter{font-weight:lighter!important}.font-weight-normal{font-weight:400!important}.font-weight-bold{font-weight:700!important}.font-weight-bolder{font-weight:bolder!important}.font-italic{font-style:italic!important}.text-white{color:#fff!important}.text-primary{color:#007bff!important}a.text-primary:focus,a.text-primary:hover{color:#0056b3!important}.text-secondary{color:#6c757d!important}a.text-secondary:focus,a.text-secondary:hover{color:#494f54!important}.text-success{color:#28a745!important}a.text-success:focus,a.text-success:hover{color:#19692c!important}.text-info{color:#17a2b8!important}a.text-info:focus,a.text-info:hover{color:#0f6674!important}.text-warning{color:#ffc107!important}a.text-warning:focus,a.text-warning:hover{color:#ba8b00!important}.text-danger{color:#dc3545!important}a.text-danger:focus,a.text-danger:hover{color:#a71d2a!important}.text-light{color:#f8f9fa!important}a.text-light:focus,a.text-light:hover{color:#cbd3da!important}.text-dark{color:#343a40!important}a.text-dark:focus,a.text-dark:hover{color:#121416!important}.text-body{color:#212529!important}.text-muted{color:#6c757d!important}.text-black-50{color:rgba(0,0,0,.5)!important}.text-white-50{color:rgba(255,255,255,.5)!important}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.text-decoration-none{text-decoration:none!important}.text-break{word-break:break-word!important;overflow-wrap:break-word!important}.text-reset{color:inherit!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}@media print{*,::after,::before{text-shadow:none!important;box-shadow:none!important}a:not(.btn){text-decoration:underline}abbr[title]::after{content:" (" attr(title) ")"}pre{white-space:pre-wrap!important}blockquote,pre{border:1px solid #adb5bd;page-break-inside:avoid}thead{display:table-header-group}img,tr{page-break-inside:avoid}h2,h3,p{orphans:3;widows:3}h2,h3{page-break-after:avoid}@page{size:a3}body{min-width:992px!important}.container{min-width:992px!important}.navbar{display:none}.badge{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered td,.table-bordered th{border:1px solid #dee2e6!important}.table-dark{color:inherit}.table-dark tbody+tbody,.table-dark td,.table-dark th,.table-dark thead th{border-color:#dee2e6}.table .thead-dark th{color:inherit;border-color:#dee2e6}}
-/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file
diff --git a/keepassxc-browser/content/autocomplete.js b/keepassxc-browser/content/autocomplete.js
index a580e2b..527faf4 100644
--- a/keepassxc-browser/content/autocomplete.js
+++ b/keepassxc-browser/content/autocomplete.js
@@ -9,7 +9,7 @@ kpxcAutocomplete.input = undefined;
kpxcAutocomplete.shadowRoot = undefined;
kpxcAutocomplete.wrapper = undefined;
-kpxcAutocomplete.create = function(input, showListInstantly = false, autoSubmit = false) {
+kpxcAutocomplete.create = async function(input, showListInstantly = false, autoSubmit = false) {
if (input.readOnly) {
return;
}
@@ -63,22 +63,20 @@ kpxcAutocomplete.showList = function(inputField) {
item.textContent += c.label;
const itemInput = kpxcUI.createElement('input', '', { 'type': 'hidden', 'value': c.value });
item.append(itemInput);
- item.addEventListener('click', function(e) {
+
+ item.addEventListener('click', async function(e) {
if (!e.isTrusted) {
return;
}
// Save index for combination.loginId
const index = Array.prototype.indexOf.call(e.currentTarget.parentElement.childNodes, e.currentTarget);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: index
- });
+ await sendMessage('page_set_login_id', index);
- inputField.value = this.getElementsByTagName('input')[0].value;
- kpxcAutocomplete.fillPassword(inputField.value, index);
+ const usernameValue = this.getElementsByTagName('input')[0].value;
+ await kpxcAutocomplete.fillPassword(usernameValue, index, c.uuid);
kpxcAutocomplete.closeList();
inputField.focus();
- document.body.removeChild(wrapper);
});
// These events prevent the double hover effect if both keyboard and mouse are used
@@ -87,6 +85,7 @@ kpxcAutocomplete.showList = function(inputField) {
item.classList.add('kpxcAutocomplete-active');
kpxcAutocomplete.index = Array.from(div.childNodes).indexOf(item);
});
+
item.addEventListener('mouseout', function(e) {
item.classList.remove('kpxcAutocomplete-active');
});
@@ -153,6 +152,7 @@ kpxcAutocomplete.getAllItems = function() {
if (!list) {
return [];
}
+
return list.getElementsByTagName('div');
};
@@ -189,7 +189,7 @@ kpxcAutocomplete.keyPress = function(e) {
if (kpxcAutocomplete.index >= 0 && items && items[kpxcAutocomplete.index] !== undefined) {
e.preventDefault();
- kpxcAutocomplete.input.value = e.currentTarget.value;
+ kpxcAutocomplete.input.value = kpxcAutocomplete.elements[kpxcAutocomplete.index].value;
kpxcAutocomplete.fillPassword(kpxcAutocomplete.input.value, kpxcAutocomplete.index);
kpxcAutocomplete.closeList();
}
@@ -212,15 +212,12 @@ kpxcAutocomplete.keyPress = function(e) {
}
};
-kpxcAutocomplete.fillPassword = async function(value, index) {
- const fieldId = kpxcAutocomplete.input.getAttribute('data-kpxc-id');
- kpxcFields.prepareId(fieldId);
-
- const givenType = kpxcAutocomplete.input.type === 'password' ? 'password' : 'username';
- const combination = await kpxcFields.getCombination(givenType, fieldId);
+kpxcAutocomplete.fillPassword = async function(value, index, uuid) {
+ const combination = await kpxcFields.getCombination(kpxcAutocomplete.input);
combination.loginId = index;
- kpxc.fillInCredentials(combination, givenType === 'password', false);
+ const manualFill = await sendMessage('page_get_manual_fill');
+ await kpxc.fillInCredentials(combination, value, uuid, manualFill === ManualFill.PASSWORD);
kpxcAutocomplete.input.setAttribute('fetched', true);
};
@@ -232,12 +229,10 @@ kpxcAutocomplete.updatePosition = function(inputField, elem) {
const rect = inputField.getBoundingClientRect();
div.style.minWidth = Pixels(inputField.offsetWidth);
- const bodyRect = document.body.getBoundingClientRect();
- const bodyStyle = getComputedStyle(document.body);
- if (bodyStyle.position.toLowerCase() === 'relative') {
- div.style.top = Pixels(rect.top - bodyRect.top + document.scrollingElement.scrollTop + inputField.offsetHeight);
- div.style.left = Pixels(rect.left - bodyRect.left + document.scrollingElement.scrollLeft);
+ if (kpxcUI.bodyStyle.position.toLowerCase() === 'relative') {
+ div.style.top = Pixels(rect.top - kpxcUI.bodyRect.top + document.scrollingElement.scrollTop + inputField.offsetHeight);
+ div.style.left = Pixels(rect.left - kpxcUI.bodyRect.left + document.scrollingElement.scrollLeft);
} else {
div.style.top = Pixels(rect.top + document.scrollingElement.scrollTop + inputField.offsetHeight);
div.style.left = Pixels(rect.left + document.scrollingElement.scrollLeft);
diff --git a/keepassxc-browser/content/banner.js b/keepassxc-browser/content/banner.js
index 0cf3b0d..eff2f3a 100644
--- a/keepassxc-browser/content/banner.js
+++ b/keepassxc-browser/content/banner.js
@@ -8,7 +8,7 @@ kpxcBanner.created = false;
kpxcBanner.credentials = {};
kpxcBanner.wrapper = undefined;
-kpxcBanner.destroy = function() {
+kpxcBanner.destroy = async function() {
kpxcBanner.created = false;
kpxcBanner.credentials = {};
@@ -17,9 +17,7 @@ kpxcBanner.destroy = function() {
kpxcBanner.banner.removeChild(dialog);
}
- browser.runtime.sendMessage({
- action: 'remove_credentials_from_tab_information'
- });
+ await sendMessage('remove_credentials_from_tab_information');
if (kpxcBanner.wrapper && window.parent.document.body.contains(kpxcBanner.wrapper)) {
window.parent.document.body.removeChild(kpxcBanner.wrapper);
@@ -29,23 +27,20 @@ kpxcBanner.destroy = function() {
};
kpxcBanner.create = async function(credentials = {}) {
- const connectedDatabase = await browser.runtime.sendMessage({
- action: 'get_connected_database'
- });
-
+ const connectedDatabase = await sendMessage('get_connected_database');
if (!kpxc.settings.showLoginNotifications || kpxcBanner.created || connectedDatabase.identifier === null) {
return;
}
// Check if database is closed
- const state = await browser.runtime.sendMessage({ action: 'check_database_hash' });
+ const state = await sendMessage('check_database_hash');
if (state === '') {
//kpxcUI.createNotification('error', tr('rememberErrorDatabaseClosed'));
return;
}
// Don't show anything if the site is in the ignore
- if (kpxc.siteIgnored(IGNORE_NORMAL)) {
+ if (await kpxc.siteIgnored(IGNORE_NORMAL)) {
return;
}
@@ -141,64 +136,48 @@ kpxcBanner.create = async function(credentials = {}) {
};
kpxcBanner.saveNewCredentials = async function(credentials = {}) {
- const result = await browser.runtime.sendMessage({
- action: 'get_database_groups'
- });
+ const result = await sendMessage('get_database_groups');
+ if (!result || !result.groups) {
+ console.log('Error: Empty result from get_database_groups');
+ return;
+ }
+
+ if (!result.defaultGroupAlwaysAsk) {
+ if (result.defaultGroup === '' || result.defaultGroup === DEFAULT_BROWSER_GROUP) {
+ // Default group is used
+ const args = [ credentials.username, credentials.password, credentials.url ];
+ const res = await sendMessage('add_credentials', args);
+ kpxcBanner.verifyResult(res);
+ return;
+ } else {
+ // A specified group is used
+ let gname = '';
+ let guuid = '';
- if (!result.defaultGroupAlwaysAsk && (result.defaultGroup !== '' && result.defaultGroup !== DEFAULT_BROWSER_GROUP)) {
- // Another group name has been specified
- const [ gname, guuid ] = kpxcBanner.getDefaultGroup(result.groups[0].children, result.defaultGroup);
- if (gname === '' && guuid === '') {
- // Root group is used -> use the root path
if (result.defaultGroup.toLowerCase() === 'root') {
result.defaultGroup = '/';
- }
-
- // Create a new group
- const newGroup = await browser.runtime.sendMessage({
- action: 'create_new_group',
- args: [ result.defaultGroup ]
- });
-
- if (newGroup.name && newGroup.uuid) {
- const res = await browser.runtime.sendMessage({
- action: 'add_credentials',
- args: [ credentials.username, credentials.password, credentials.url, newGroup.name, newGroup.uuid ]
- });
- kpxcBanner.verifyResult(res);
+ gname = result.groups[0].name;
+ guuid = result.groups[0].uuid;
} else {
- kpxcUI.createNotification('error', tr('rememberErrorCreatingNewGroup'));
+ [ gname, guuid ] = kpxcBanner.getDefaultGroup(result.groups[0].children, result.defaultGroup);
+ if (gname === '' && guuid === '') {
+ // Create a new group
+ const newGroup = await sendMessage('create_new_group', [ result.defaultGroup ]);
+ if (newGroup.name && newGroup.uuid) {
+ const res = await sendMessage('add_credentials', [ credentials.username, credentials.password, credentials.url, newGroup.name, newGroup.uuid ]);
+ kpxcBanner.verifyResult(res);
+ } else {
+ kpxcUI.createNotification('error', tr('rememberErrorCreatingNewGroup'));
+ }
+
+ return;
+ }
}
- return;
- }
- const res = await browser.runtime.sendMessage({
- action: 'add_credentials',
- args: [ credentials.username, credentials.password, credentials.url, gname, guuid ]
- });
- kpxcBanner.verifyResult(res);
- return;
- } else if ((result.groups === undefined || (result.groups.length > 0 && result.groups[0].children.length === 0))
- || (!result.defaultGroupAlwaysAsk && (result.defaultGroup === '' || result.defaultGroup === DEFAULT_BROWSER_GROUP))) {
- // Only the Root group and no KeePassXC-Browser passwords -> save to default
- // Or when default group is not set and defaultGroupAskAlways is disabled -> save to default
- const args = [ credentials.username, credentials.password, credentials.url ];
-
- // If root group is defined by the user, and there's no default browser group, save the credentials to the root group
- if (result.groups !== undefined
- && result.groups.length > 0
- && result.groups[0].children.length === 0
- && (result.defaultGroup.toLowerCase() === 'root'
- || result.defaultGroup === '/')) {
- args.push(result.groups[0].name, result.groups[0].uuid);
+ const res = await sendMessage('add_credentials', [ credentials.username, credentials.password, credentials.url, gname, guuid ]);
+ kpxcBanner.verifyResult(res);
+ return;
}
-
- const res = await browser.runtime.sendMessage({
- action: 'add_credentials',
- args: args
- });
- kpxcBanner.verifyResult(res);
- return;
}
const addChildren = function(group, parentElement, depth) {
@@ -227,11 +206,7 @@ kpxcBanner.saveNewCredentials = async function(credentials = {}) {
return;
}
- const res = await browser.runtime.sendMessage({
- action: 'add_credentials',
- args: [ credentials.username, credentials.password, credentials.url, group, groupUuid ]
- });
-
+ const res = await sendMessage('add_credentials', [ credentials.username, credentials.password, credentials.url, group, groupUuid ]);
kpxcBanner.verifyResult(res);
});
@@ -264,10 +239,7 @@ kpxcBanner.updateCredentials = async function(credentials = {}) {
credentials.username = credentials.list[0].login;
}
- const res = await browser.runtime.sendMessage({
- action: 'update_credentials',
- args: [ credentials.list[0].uuid, credentials.username, credentials.password, credentials.url ]
- });
+ const res = await sendMessage('update_credentials', [ credentials.list[0].uuid, credentials.username, credentials.password, credentials.url ]);
kpxcBanner.verifyResult(res);
} else {
await kpxcBanner.createCredentialDialog();
@@ -310,10 +282,7 @@ kpxcBanner.updateCredentials = async function(credentials = {}) {
return;
}
- const res = await browser.runtime.sendMessage({
- action: 'update_credentials',
- args: [ credentials.list[entryId].uuid, credentials.username, credentials.password, credentials.url ]
- });
+ const res = await sendMessage('update_credentials', [ credentials.list[entryId].uuid, credentials.username, credentials.password, credentials.url ]);
kpxcBanner.verifyResult(res);
});
});
@@ -369,9 +338,7 @@ kpxcBanner.createCredentialDialog = async function() {
kpxcBanner.shadowSelector('#kpxc-banner-btn-update').hidden = true;
kpxcBanner.shadowSelector('.kpxc-checkbox').disabled = true;
- const connectedDatabase = await browser.runtime.sendMessage({
- action: 'get_connected_database'
- });
+ const connectedDatabase = await sendMessage('get_connected_database');
const databaseName = connectedDatabase.count > 0 ? connectedDatabase.identifier : '';
const dialog = kpxcUI.createElement('div', 'kpxc-banner-dialog');
diff --git a/keepassxc-browser/content/define.js b/keepassxc-browser/content/define.js
index de78d1d..196efa8 100644
--- a/keepassxc-browser/content/define.js
+++ b/keepassxc-browser/content/define.js
@@ -8,15 +8,18 @@ kpxcDefine.selection = {
totp: null,
fields: []
};
-kpxcDefine.eventFieldClick = null;
+
kpxcDefine.dialog = null;
-kpxcDefine.startPosX = 0;
-kpxcDefine.startPosY = 0;
kpxcDefine.diffX = 0;
kpxcDefine.diffY = 0;
+kpxcDefine.eventFieldClick = null;
+kpxcDefine.inputQueryPattern = 'input[type=\'text\'], input[type=\'email\'], input[type=\'password\'], input[type=\'tel\'], input[type=\'number\'], input[type=\'username\'], input:not([type])';
+kpxcDefine.markedFields= [];
kpxcDefine.keyDown = null;
+kpxcDefine.startPosX = 0;
+kpxcDefine.startPosY = 0;
-kpxcDefine.init = function() {
+kpxcDefine.init = async function() {
const backdrop = kpxcUI.createElement('div', 'kpxcDefine-modal-backdrop', { 'id': 'kpxcDefine-backdrop' });
const chooser = kpxcUI.createElement('div', '', { 'id': 'kpxcDefine-fields' });
const description = kpxcUI.createElement('div', '', { 'id': 'kpxcDefine-description' });
@@ -25,9 +28,6 @@ kpxcDefine.init = function() {
document.body.append(backdrop);
document.body.append(chooser);
- kpxcFields.getAllFields();
- kpxcFields.prepareVisibleFieldsWithID('select');
-
kpxcDefine.initDescription();
kpxcDefine.resetSelection();
kpxcDefine.prepareStep1();
@@ -117,21 +117,24 @@ kpxcDefine.resetSelection = function() {
fields: []
};
+ kpxcDefine.markedFields = [];
+
const fields = $('#kpxcDefine-fields');
if (fields) {
fields.textContent = '';
}
};
-kpxcDefine.isFieldSelected = function(kpxcId) {
- if (kpxcId) {
+kpxcDefine.isFieldSelected = function(field) {
+ if (kpxcDefine.markedFields.some(f => f === field)) {
return (
- kpxcId === kpxcDefine.selection.username
- || kpxcId === kpxcDefine.selection.password
- || kpxcId === kpxcDefine.selection.totp
- || kpxcId in kpxcDefine.selection.fields
+ (kpxcDefine.selection.username && kpxcDefine.selection.username.originalElement === field)
+ || (kpxcDefine.selection.password && kpxcDefine.selection.password.originalElement === field)
+ || (kpxcDefine.selection.totp && kpxcDefine.selection.totp.originalElement === field)
+ || kpxcDefine.selection.fields.includes(field)
);
}
+
return false;
};
@@ -142,35 +145,37 @@ kpxcDefine.markAllUsernameFields = function(chooser) {
}
const field = elem || e.currentTarget;
- kpxcDefine.selection.username = field.getAttribute('data-kpxc-id');
field.classList.add('kpxcDefine-fixed-username-field');
field.textContent = tr('username');
field.onclick = null;
+ kpxcDefine.selection.username = field;
+ kpxcDefine.markedFields.push(field.originalElement);
+
kpxcDefine.prepareStep2();
kpxcDefine.markAllPasswordFields('#kpxcDefine-fields');
};
- kpxcDefine.markFields(chooser, kpxcFields.inputQueryPattern);
+
+ kpxcDefine.markFields(chooser, kpxcDefine.inputQueryPattern);
};
-kpxcDefine.markAllPasswordFields = function(chooser, more = false) {
+kpxcDefine.markAllPasswordFields = function(chooser) {
kpxcDefine.eventFieldClick = function(e, elem) {
if (!e.isTrusted) {
return;
}
const field = elem || e.currentTarget;
- kpxcDefine.selection.password = field.getAttribute('data-kpxc-id');
field.classList.add('kpxcDefine-fixed-password-field');
field.textContent = tr('password');
field.onclick = null;
+ kpxcDefine.selection.password = field;
+ kpxcDefine.markedFields.push(field.originalElement);
+
kpxcDefine.prepareStep3();
kpxcDefine.markAllTOTPFields('#kpxcDefine-fields');
};
- if (more) {
- kpxcDefine.markFields(chooser, kpxcFields.inputQueryPattern);
- } else {
- kpxcDefine.markFields(chooser, 'input[type=\'password\']');
- }
+
+ kpxcDefine.markFields(chooser, 'input[type=\'password\']');
};
kpxcDefine.markAllStringFields = function(chooser) {
@@ -180,15 +185,19 @@ kpxcDefine.markAllStringFields = function(chooser) {
}
const field = elem || e.currentTarget;
- const value = field.getAttribute('data-kpxc-id');
- kpxcDefine.selection.fields[value] = true;
+ if (kpxcDefine.isFieldSelected(field.originalElement)) {
+ return;
+ }
+
+ kpxcDefine.selection.fields.push(field.originalElement);
+ kpxcDefine.markedFields.push(field.originalElement);
- const count = Object.keys(kpxcDefine.selection.fields).length;
field.classList.add('kpxcDefine-fixed-string-field');
- field.textContent = tr('defineStringField') + String(count);
+ field.textContent = tr('defineStringField') + String(kpxcDefine.selection.fields.length);
field.onclick = null;
};
- kpxcDefine.markFields(chooser, kpxcFields.inputQueryPattern + ', select');
+
+ kpxcDefine.markFields(chooser, kpxcDefine.inputQueryPattern + ', select');
};
kpxcDefine.markAllTOTPFields = function(chooser) {
@@ -198,14 +207,17 @@ kpxcDefine.markAllTOTPFields = function(chooser) {
}
const field = elem || e.currentTarget;
- kpxcDefine.selection.totp = field.getAttribute('data-kpxc-id');
field.classList.add('kpxcDefine-fixed-totp-field');
field.textContent = 'TOTP';
field.onclick = null;
+ kpxcDefine.selection.totp = field;
+ kpxcDefine.markedFields.push(field.originalElement);
+
kpxcDefine.prepareStep4();
kpxcDefine.markAllStringFields('#kpxcDefine-fields');
};
- kpxcDefine.markFields(chooser, kpxcFields.inputQueryPattern);
+
+ kpxcDefine.markFields(chooser, kpxcDefine.inputQueryPattern);
};
kpxcDefine.markFields = function(chooser, pattern) {
@@ -214,39 +226,49 @@ kpxcDefine.markFields = function(chooser, pattern) {
const inputs = document.querySelectorAll(pattern);
for (const i of inputs) {
- if (kpxcDefine.isFieldSelected(i.getAttribute('data-kpxc-id'))) {
+ if (kpxcDefine.isFieldSelected(i)) {
+ continue;
+ }
+
+ if (!kpxcFields.isVisible(i)) {
continue;
}
- if (kpxcFields.isVisible(i)) {
- const field = kpxcUI.createElement('div', 'kpxcDefine-fixed-field', { 'data-kpxc-id': i.getAttribute('data-kpxc-id') });
- const rect = i.getBoundingClientRect();
- field.style.top = Pixels(rect.top);
- field.style.left = Pixels(rect.left);
- field.style.width = Pixels(rect.width);
- field.style.height = Pixels(rect.height);
- field.textContent = String(index);
- field.addEventListener('click', function(e) {
- kpxcDefine.eventFieldClick(e);
- });
- field.addEventListener('mouseenter', function() {
- field.classList.add('kpxcDefine-fixed-hover-field');
- });
- field.addEventListener('mouseleave', function() {
- field.classList.remove('kpxcDefine-fixed-hover-field');
- });
- i.addEventListener('focus', function() {
- field.classList.add('kpxcDefine-fixed-hover-field');
- });
- i.addEventListener('blur', function() {
- field.classList.remove('kpxcDefine-fixed-hover-field');
- });
- const elem = $(chooser);
- if (elem) {
- elem.append(field);
- firstInput = field;
- ++index;
- }
+ const field = kpxcUI.createElement('div', 'kpxcDefine-fixed-field');
+ field.originalElement = i;
+
+ const rect = i.getBoundingClientRect();
+ field.style.top = Pixels(rect.top);
+ field.style.left = Pixels(rect.left);
+ field.style.width = Pixels(rect.width);
+ field.style.height = Pixels(rect.height);
+ field.textContent = String(index);
+
+ field.addEventListener('click', function(e) {
+ kpxcDefine.eventFieldClick(e);
+ });
+
+ field.addEventListener('mouseenter', function() {
+ field.classList.add('kpxcDefine-fixed-hover-field');
+ });
+
+ field.addEventListener('mouseleave', function() {
+ field.classList.remove('kpxcDefine-fixed-hover-field');
+ });
+
+ i.addEventListener('focus', function() {
+ field.classList.add('kpxcDefine-fixed-hover-field');
+ });
+
+ i.addEventListener('blur', function() {
+ field.classList.remove('kpxcDefine-fixed-hover-field');
+ });
+
+ const elem = $(chooser);
+ if (elem) {
+ elem.append(field);
+ firstInput = field;
+ ++index;
}
}
@@ -298,7 +320,7 @@ kpxcDefine.prepareStep4 = function() {
$('#kpxcDefine-help').style.marginBottom = '10px';
$('#kpxcDefine-help').textContent = tr('defineHelpText');
- removeContent('div.kpxcDefine-fixed-field:not(.kpxcDefine-fixed-username-field):not(.kpxcDefine-fixed-password-field):not(.kpxcDefine-fixed-totp-field)');
+ removeContent('div.kpxcDefine-fixed-field:not(.kpxcDefine-fixed-username-field):not(.kpxcDefine-fixed-password-field):not(.kpxcDefine-fixed-totp-field):not(.kpxcDefine-fixed-string-field)');
$('#kpxcDefine-chooser-headline').textContent = tr('defineConfirmSelection');
kpxcDefine.dataStep = 4;
$('#kpxcDefine-btn-skip').style.display = 'none';
@@ -330,10 +352,17 @@ kpxcDefine.again = function() {
};
kpxcDefine.more = function() {
- if (kpxcDefine.dataStep === 2) {
+ if (kpxcDefine.dataStep === 1) {
+ kpxcDefine.prepareStep1();
+ } else if (kpxcDefine.dataStep === 2) {
kpxcDefine.prepareStep2();
- kpxcDefine.markAllPasswordFields('#kpxcDefine-fields', true);
+ } else if (kpxcDefine.dataStep === 3) {
+ kpxcDefine.prepareStep3();
+ } else if (kpxcDefine.dataStep === 4) {
+ kpxcDefine.prepareStep4();
}
+
+ kpxcDefine.markFields('#kpxcDefine-fields', kpxcDefine.inputQueryPattern + ', select');
};
kpxcDefine.confirm = async function() {
@@ -346,21 +375,20 @@ kpxcDefine.confirm = async function() {
}
if (kpxcDefine.selection.username) {
- kpxcDefine.selection.username = kpxcFields.prepareId(kpxcDefine.selection.username);
+ kpxcDefine.selection.username = kpxcFields.getId(kpxcDefine.selection.username.originalElement);
}
if (kpxcDefine.selection.password) {
- kpxcDefine.selection.password = kpxcFields.prepareId(kpxcDefine.selection.password);
+ kpxcDefine.selection.password = kpxcFields.getId(kpxcDefine.selection.password.originalElement);
}
if (kpxcDefine.selection.totp) {
- kpxcDefine.selection.totp = kpxcFields.prepareId(kpxcDefine.selection.totp);
+ kpxcDefine.selection.totp = kpxcFields.getId(kpxcDefine.selection.totp.originalElement);
}
const fieldIds = [];
- const fieldKeys = Object.keys(kpxcDefine.selection.fields);
- for (const i of fieldKeys) {
- fieldIds.push(kpxcFields.prepareId(i));
+ for (const i of kpxcDefine.selection.fields) {
+ fieldIds.push(kpxcFields.getId(i));
}
const location = kpxc.getDocumentLocation();
@@ -371,11 +399,7 @@ kpxcDefine.confirm = async function() {
fields: fieldIds
};
- await browser.runtime.sendMessage({
- action: 'save_settings',
- args: [ kpxc.settings ]
- });
-
+ await sendMessage('save_settings', kpxc.settings);
kpxcDefine.close();
};
@@ -387,19 +411,13 @@ kpxcDefine.discard = async function() {
const location = kpxc.getDocumentLocation();
delete kpxc.settings['defined-custom-fields'][location];
- await browser.runtime.sendMessage({
- action: 'save_settings',
- args: [ kpxc.settings ]
- });
-
- await browser.runtime.sendMessage({
- action: 'load_settings'
- });
+ await sendMessage('save_settings', kpxc.settings);
+ await sendMessage('load_settings');
$('div.alreadySelected').remove();
};
-// Handle the keyboard events
+// Handle keyboard events
kpxcDefine.keyDown = function(e) {
if (!e.isTrusted) {
return;
@@ -413,7 +431,7 @@ kpxcDefine.keyDown = function(e) {
// Select input field by number
e.preventDefault();
const index = e.keyCode - 48;
- const inputFields = document.querySelectorAll('div.kpxcDefine-fixed-field:not(.kpxcDefine-fixed-username-field):not(.kpxcDefine-fixed-password-field)');
+ const inputFields = document.querySelectorAll('div.kpxcDefine-fixed-field:not(.kpxcDefine-fixed-username-field):not(.kpxcDefine-fixed-password-field):not(.kpxcDefine-fixed-totp-field)');
if (inputFields.length >= index) {
kpxcDefine.eventFieldClick(e, inputFields[index - 1]);
diff --git a/keepassxc-browser/content/keepassxc-browser.js b/keepassxc-browser/content/keepassxc-browser.js
index 489cf79..353753b 100755
--- a/keepassxc-browser/content/keepassxc-browser.js
+++ b/keepassxc-browser/content/keepassxc-browser.js
@@ -1,144 +1,223 @@
'use strict';
-const ManualFill = {
- NONE: 0,
- PASS: 1,
- BOTH: 2
-};
-
-const DatabaseState = {
- DISCONNECTED: 0,
- LOCKED: 1,
- UNLOCKED: 2
-};
-
-const acceptedOTPFields = [
- '2fa',
- 'auth',
- 'challenge',
- 'code',
- 'mfa',
- 'otp',
- 'token'
-];
+const _maximumInputs = 100;
+const _maximumMutations = 200;
// Contains already called method names
const _called = {};
-_called.retrieveCredentials = false;
_called.clearLogins = false;
-_called.manualFillRequested = ManualFill.NONE;
-let _singleInputEnabledForPage = false;
-let _databaseState = DatabaseState.DISCONNECTED;
-const _maximumInputs = 100;
-const _maximumMutations = 200;
+_called.retrieveCredentials = false;
-// Count of detected form fields on the page
-var _detectedFields = 0;
+// Wrapper
+const sendMessage = async function(action, args) {
+ return await browser.runtime.sendMessage({ action: action, args: args });
+};
+
+/**
+ * @Object kpxcIcons
+ * Icon handling.
+ */
+const kpxcIcons = {};
+kpxcIcons.icons = [];
+kpxcIcons.iconTypes = { USERNAME: 0, PASSWORD: 1, TOTP: 2 };
-// Element id's containing input fields detected by MutationObserver
-const _observerIds = [];
+// Adds an icon to input field
+kpxcIcons.addIcon = async function(field, iconType) {
+ if (!field || iconType < 0 || iconType > 2) {
+ return;
+ }
-// Document URL
-let _documentURL = document.location.href;
+ let iconSet = false;
+ if (iconType === kpxcIcons.iconTypes.USERNAME && kpxcUsernameIcons.isValid(field)) {
+ kpxcUsernameIcons.newIcon(field, kpxc.databaseState);
+ iconSet = true;
+ } else if (iconType === kpxcIcons.iconTypes.PASSWORD && kpxcPasswordIcons.isValid(field)) {
+ kpxcPasswordIcons.newIcon(field, kpxc.databaseState);
+ iconSet = true;
+ } else if (iconType === kpxcIcons.iconTypes.TOTP && kpxcTOTPIcons.isValid(field)) {
+ kpxcTOTPIcons.newIcon(field, kpxc.databaseState);
+ iconSet = true;
+ }
-// These are executed in each frame
-browser.runtime.onMessage.addListener(async function(req, sender) {
- if ('action' in req) {
- // Don't allow any actions if the site is ignored
- if (kpxc.siteIgnored()) {
- return Promise.resolve();
+ if (iconSet) {
+ kpxcIcons.icons.push({
+ field: field,
+ iconType: iconType
+ });
+ }
+};
+
+// Adds all necessary icons to a saved form
+kpxcIcons.addIconsFromForm = async function(form) {
+ const addUsernameIcons = async function(c) {
+ if (kpxc.settings.showLoginFormIcon && await kpxc.passwordFilled() === false) {
+ // Special case where everything else has been hidden, but a single password field is now displayed.
+ // For example PayPal and Amazon is handled like this.
+ if (c.username && !c.password && c.passwordInputs.length === 1) {
+ kpxcIcons.addIcon(c.passwordInputs[0], kpxcIcons.iconTypes.USERNAME);
+ }
+
+ if (c.username && !c.username.readOnly) {
+ kpxcIcons.addIcon(c.username, kpxcIcons.iconTypes.USERNAME);
+ } else if (c.password && (!c.username || (c.username && c.username.readOnly))) {
+ // Single password field
+ kpxcIcons.addIcon(c.password, kpxcIcons.iconTypes.USERNAME);
+ }
}
+ };
- if (req.action === 'fill_user_pass_with_specific_login') {
- kpxc.fillWithSpecificLogin(req.id);
- } else if (req.action === 'fill_username_password') {
- _called.manualFillRequested = ManualFill.BOTH;
- await kpxc.receiveCredentialsIfNecessary();
- kpxc.fillInFromActiveElement(false);
- } else if (req.action === 'fill_password') {
- _called.manualFillRequested = ManualFill.PASS;
- await kpxc.receiveCredentialsIfNecessary();
- kpxc.fillInFromActiveElement(false, true); // passOnly to true
- } else if (req.action === 'fill_totp') {
- await kpxc.receiveCredentialsIfNecessary();
- kpxc.fillInFromActiveElementTOTPOnly();
- } else if (req.action === 'clear_credentials') {
- kpxcEvents.clearCredentials();
- return Promise.resolve();
- } else if (req.action === 'activated_tab') {
- kpxcEvents.triggerActivatedTab();
- return Promise.resolve();
- } else if (req.action === 'ignore_site') {
- kpxc.ignoreSite(req.args);
- } else if (req.action === 'check_database_hash' && 'hash' in req) {
- kpxc.detectDatabaseChange(req);
- } else if (req.action === 'remember_credentials') {
- kpxc.contextMenuRememberCredentials();
- } else if (req.action === 'choose_credential_fields') {
- kpxcDefine.init();
- } else if (req.action === 'redetect_fields') {
- const response = await browser.runtime.sendMessage({
- action: 'load_settings'
- });
- kpxc.settings = response;
- kpxc.initCredentialFields(true);
- } else if (req.action === 'show_password_generator') {
- kpxcPasswordDialog.trigger();
- } else if (req.action === 'add_username_only_option') {
- kpxc.addToSitePreferences();
+ const addPasswordIcons = async function(c) {
+ // Show password icons also with forms without any username field
+ if (kpxc.settings.usePasswordGeneratorIcons
+ && ((c.username && c.password) || (!c.username && c.passwordInputs.length > 0))) {
+ for (const input of c.passwordInputs) {
+ kpxcIcons.addIcon(input, kpxcIcons.iconTypes.PASSWORD);
+ }
+ }
+ };
+
+ const addTOTPIcons = async function(c) {
+ if (c.totp && kpxc.settings.showOTPIcon) {
+ kpxcIcons.addIcon(c.totp, kpxcIcons.iconTypes.TOTP);
}
+ };
+
+ await Promise.all([
+ await addUsernameIcons(form),
+ await addPasswordIcons(form),
+ await addTOTPIcons(form)
+ ]);
+};
+
+// Delete all icons that have been hidden from the page view
+kpxcIcons.deleteHiddenIcons = function() {
+ kpxcUsernameIcons.deleteHiddenIcons();
+ kpxcPasswordIcons.deleteHiddenIcons();
+ kpxcTOTPIcons.deleteHiddenIcons();
+};
+
+// Initializes all icons needed to be shown
+kpxcIcons.initIcons = async function(combinations = []) {
+ if (combinations.length === 0) {
+ return;
}
-});
-function _f(fieldId) {
- const inputs = document.querySelectorAll(`input[data-kpxc-id='${fieldId}']`);
- return inputs.length > 0 ? inputs[0] : null;
-}
+ for (const form of kpxcForm.savedForms) {
+ await kpxcIcons.addIconsFromForm(form);
+ }
-function _fs(fieldId) {
- const inputs = document.querySelectorAll(`input[data-kpxc-id='${fieldId}'], select[data-kpxc-id='${fieldId}']`);
- return inputs.length > 0 ? inputs[0] : null;
-}
+ // Check for other combinations that are not in any form
+ for (const c of combinations) {
+ if (c.form) {
+ continue;
+ }
+ await kpxcIcons.addIconsFromForm(c);
+ }
+};
+
+kpxcIcons.hasIcon = function(field) {
+ return !field ? false : kpxcIcons.icons.some(i => i.field === field);
+};
+// Sets the icons to corresponding database lock status
+kpxcIcons.switchIcons = function() {
+ kpxcUsernameIcons.switchIcon(kpxc.databaseState);
+ kpxcPasswordIcons.switchIcon(kpxc.databaseState);
+ kpxcTOTPIcons.switchIcon(kpxc.databaseState);
+};
+
+
+/**
+ * @Object kpxcForm
+ * Identifies form submits and password changes.
+ */
const kpxcForm = {};
-kpxcForm.passwordInputs = [];
+kpxcForm.formButtonQuery = 'button[type=\'button\'], button[type=\'submit\'], input[type=\'button\'], button:not([type]), div[role=\'button\']';
+kpxcForm.savedForms = [];
-kpxcForm.init = function(form, credentialFields) {
- if (!form.getAttribute('kpxcForm-initialized') && (credentialFields.password || credentialFields.username)) {
- form.setAttribute('kpxcForm-initialized', true);
- kpxcForm.setInputFields(form, credentialFields);
- form.addEventListener('submit', kpxcForm.onSubmit);
+// Returns true if form has been already saved
+kpxcForm.formIdentified = function(form) {
+ return kpxcForm.savedForms.some(f => f.form === form);
+};
- const submitButton = kpxc.getFormSubmitButton(form);
- if (submitButton !== undefined) {
- submitButton.addEventListener('click', kpxcForm.onSubmit);
+// Return input fields from our Object array
+kpxcForm.getCredentialFieldsFromForm = function(form) {
+ for (const savedForm of kpxcForm.savedForms) {
+ if (savedForm.form === form) {
+ return [ savedForm.username, savedForm.password, savedForm.passwordInputs, savedForm.totp ];
}
}
+
+ return [];
};
-kpxcForm.destroy = function(form, credentialFields) {
- if (form === false && credentialFields) {
- const field = _f(credentialFields.password) || _f(credentialFields.username);
- if (field) {
- form = kpxc.getForm(field);
- if (form && form.length > 0) {
- form.removeEventListener('submit', kpxcForm.onSubmit);
+// Get the form submit button instead if action URL is same as the page itself
+kpxcForm.getFormSubmitButton = function(form) {
+ const action = kpxc.submitUrl || form.action;
+ if (action.includes(document.location.origin + document.location.pathname)) {
+ for (const i of form.elements) {
+ if (i.type === 'submit') {
+ return i;
}
}
}
- kpxcForm.passwordInputs = [];
+ // Try to find another button. Select the first one.
+ const buttons = Array.from(form.querySelectorAll(kpxcForm.formButtonQuery));
+ if (buttons.length > 0) {
+ return buttons[0];
+ }
+
+ // Try to find similar buttons outside the form which are added via 'form' property
+ for (const e of form.elements) {
+ if ((e.nodeName === 'BUTTON' && (e.type === 'button' || e.type === 'submit' || e.type === ''))
+ || (e.nodeName === 'INPUT' && e.type === 'button')) {
+ return e;
+ }
+ }
+
+ return undefined;
};
-kpxcForm.setInputFields = function(form, credentialFields) {
- form.setAttribute('kpxcUsername', credentialFields.username);
- form.setAttribute('kpxcPassword', credentialFields.password);
+// Retrieve new password from a form with three elements: Current, New, Repeat New
+kpxcForm.getNewPassword = function(passwordInputs = []) {
+ if (passwordInputs.length < 2) {
+ return '';
+ }
- // Save all the password input fields from the form
- kpxcForm.passwordInputs = Array.from(form.elements).filter(e => e.nodeName === 'INPUT' && e.type === 'password');
+ // Just two password fields, current and new
+ if (passwordInputs.length === 2 && passwordInputs[0] !== passwordInputs[1]) {
+ return passwordInputs[1].value;
+ }
+
+ // Choose the last three password fields. The first ones are almost always for something else
+ const current = passwordInputs[passwordInputs.length - 3].value;
+ const newPass = passwordInputs[passwordInputs.length - 2].value;
+ const repeatNew = passwordInputs[passwordInputs.length - 1].value;
+
+ if ((newPass === repeatNew && current !== newPass && current !== repeatNew)
+ || (current === newPass && repeatNew !== newPass && repeatNew !== current)) {
+ return newPass;
+ }
+
+ return '';
};
+// Initializes form and attaches the submit button to our own callback
+kpxcForm.init = function(form, credentialFields) {
+ if (!kpxcForm.formIdentified(form) && (credentialFields.password || credentialFields.username)) {
+ kpxcForm.saveForm(form, credentialFields);
+ form.addEventListener('submit', kpxcForm.onSubmit);
+
+ const submitButton = kpxcForm.getFormSubmitButton(form);
+ if (submitButton !== undefined) {
+ submitButton.addEventListener('click', kpxcForm.onSubmit);
+ }
+ }
+};
+
+// Triggers when form is submitted. Shows the credential banner
kpxcForm.onSubmit = async function(e) {
if (!e.isTrusted) {
return;
@@ -156,10 +235,7 @@ kpxcForm.onSubmit = async function(e) {
return;
}
- const usernameId = form.getAttribute('kpxcUsername');
- const passwordId = form.getAttribute('kpxcPassword');
- const usernameField = _f(usernameId);
- const passwordField = _f(passwordId);
+ const [ usernameField, passwordField, passwordInputs ] = kpxcForm.getCredentialFieldsFromForm(form);
let usernameValue = '';
let passwordValue = '';
@@ -171,8 +247,8 @@ kpxcForm.onSubmit = async function(e) {
}
// Check if the form has three password fields -> a possible password change form
- if (kpxcForm.passwordInputs.length >= 2) {
- passwordValue = kpxcForm.getNewPassword();
+ if (passwordInputs && passwordInputs.length >= 2) {
+ passwordValue = kpxcForm.getNewPassword(passwordInputs);
} else if (passwordField) {
// Use the combination password field instead
passwordValue = passwordField.value;
@@ -183,114 +259,181 @@ kpxcForm.onSubmit = async function(e) {
return;
}
- browser.runtime.sendMessage({
- action: 'page_set_submitted',
- args: [ true, usernameValue, passwordValue, trimURL(window.top.location.href), kpxc.credentials ]
- });
+ await kpxc.setPasswordFilled(true);
+ await sendMessage('page_set_submitted', [ true, usernameValue, passwordValue, trimURL(window.top.location.href), kpxc.credentials ]);
// Show the banner if the page does not reload
kpxc.rememberCredentials(usernameValue, passwordValue);
};
-// Retrieve new password from a form with three elements: Current, New, Repeat New
-kpxcForm.getNewPassword = function() {
- if (kpxcForm.passwordInputs.length < 2) {
- return '';
- }
+// Save form to Object array
+kpxcForm.saveForm = function(form, combination) {
+ kpxcForm.savedForms.push({
+ form: form,
+ username: combination.username,
+ password: combination.password,
+ totp: combination.totp,
+ passwordInputs: Array.from(form.elements).filter(e => e.nodeName === 'INPUT' && e.type === 'password')
+ });
+};
- // Just two password fields, current and new
- if (kpxcForm.passwordInputs.length === 2
- && kpxcForm.passwordInputs[0] !== kpxcForm.passwordInputs[1]) {
- return kpxcForm.passwordInputs[1].value;
+
+/**
+ * @Object kpxcFields
+ * Provides methods for input field handling.
+ */
+const kpxcFields = {};
+
+// Returns all username & password combinations detected from the inputs.
+// After username field is detected, first password field found after that will be saved as a combination.
+kpxcFields.getAllCombinations = async function(inputs) {
+ const combinations = [];
+ let usernameField = null;
+
+ for (const input of inputs) {
+ if (!input) {
+ continue;
+ }
+
+ if (input.getLowerCaseAttribute('type') === 'password') {
+ const combination = {
+ username: (!usernameField || usernameField.size < 1) ? null : usernameField,
+ password: input,
+ passwordInputs: [ input ],
+ form: input.form
+ };
+
+ combinations.push(combination);
+ usernameField = null;
+ } else if (kpxcTOTPIcons.isValid(input)) {
+ // Dynamically added TOTP field
+ const combination = {
+ username: null,
+ password: null,
+ passwordInputs: [],
+ totp: input,
+ form: null
+ };
+
+ combinations.push(combination);
+ } else {
+ usernameField = input;
+ }
}
- // Choose the last three password fields. The first ones are almost always for something else
- const current = kpxcForm.passwordInputs[kpxcForm.passwordInputs.length - 3].value;
- const newPass = kpxcForm.passwordInputs[kpxcForm.passwordInputs.length - 2].value;
- const repeatNew = kpxcForm.passwordInputs[kpxcForm.passwordInputs.length - 1].value;
+ if (kpxc.singleInputEnabledForPage && combinations.length === 0 && usernameField) {
+ const combination = {
+ username: usernameField,
+ password: null,
+ passwordInputs: [],
+ form: usernameField.form
+ };
- if (current !== newPass && newPass !== '' && newPass === repeatNew) {
- return newPass;
- } else if (current === newPass && repeatNew !== newPass) {
- // Reverse form where current password is at the bottom
- return repeatNew;
+ combinations.push(combination);
}
- return '';
+ return combinations;
};
+// Return all input fields on the page, but ignore previously detected
+kpxcFields.getAllPageInputs = async function(previousInputs = []) {
+ const fields = [];
+ const inputs = kpxcObserverHelper.getInputs(document);
-const kpxcFields = {};
-kpxcFields.inputQueryPattern = 'input[type=\'text\'], input[type=\'email\'], input[type=\'password\'], input[type=\'tel\'], input[type=\'number\'], input[type=\'username\'], input:not([type])';
+ for (const input of inputs) {
+ // Ignore fields that are already detected
+ if (previousInputs.some(e => e === input)) {
+ continue;
+ }
-// copied from Sizzle.js
-kpxcFields.rcssescape = /([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g;
-kpxcFields.fcssescape = function(ch, asCodePoint) {
- if (asCodePoint) {
- // U+0000 NULL becomes U+FFFD REPLACEMENT CHARACTER
- if (ch === '\0') {
- return '\uFFFD';
+ if (kpxcFields.isVisible(input) && !kpxcFields.isSearchField(input) && kpxcFields.isAutocompleteAppropriate(input)) {
+ fields.push(input);
}
+ }
- // Control characters and (dependent upon position) numbers get escaped as code points
- return ch.slice(0, -1) + '\\' + ch.charCodeAt(ch.length - 1).toString(16) + ' ';
+ kpxc.detectedFields = previousInputs.length + fields.length;
+
+ // Show add username-only option for the site in popup
+ if (!kpxc.singleInputEnabledForPage
+ && ((fields.length === 1 && fields[0].getLowerCaseAttribute('type') !== 'password')
+ || (previousInputs.length === 1 && previousInputs[0].getLowerCaseAttribute('type') !== 'password'))) {
+ sendMessage('username_field_detected', true );
+ } else {
+ sendMessage('username_field_detected', false );
}
- // Other potentially-special ASCII characters get backslash-escaped
- return '\\' + ch;
+ await kpxc.initCombinations(inputs);
+ return fields;
};
-// Unique number as new IDs for input fields
-kpxcFields.uniqueNumber = 342845638;
-// Objects with combination of username + password fields
-kpxcFields.combinations = [];
-
-kpxcFields.setUniqueId = function(field) {
- if (field && !field.getAttribute('data-kpxc-id')) {
- // Use ID of field if it is unique
- const fieldId = field.getAttribute('id');
- if (fieldId) {
- const foundIds = document.querySelectorAll('input#' + kpxcFields.prepareId(fieldId));
- if (foundIds.length === 1) {
- field.setAttribute('data-kpxc-id', fieldId);
- return;
+/**
+ * Returns the combination where input field is used
+ * @param {HTMLElement} field Input field
+ * @param {String} givenType 'username' or 'password'
+ */
+kpxcFields.getCombination = async function(field, givenType) {
+ // If givenType is not set, return the combination that uses the selected field
+ for (const combination of kpxc.combinations) {
+ if (!givenType && Object.values(combination).find(c => c === field)) {
+ return combination;
+ } else if (givenType) {
+ if (combination[givenType] === field) {
+ return combination;
}
}
+ }
- // Create own ID if no ID is set for this field
- kpxcFields.uniqueNumber += 1;
- field.setAttribute('data-kpxc-id', 'kpxcpw' + String(kpxcFields.uniqueNumber));
+ return undefined;
+};
+
+// Gets of generates an unique ID for the element
+kpxcFields.getId = function(target) {
+ if (target.classList.length > 0) {
+ return `${target.nodeName} ${target.type} ${target.classList.value}`;
+ }
+
+ if (target.id && target.id !== '') {
+ return `${target.nodeName} ${target.type} ${kpxcFields.prepareId(target.id)}`;
}
+
+ return `kpxc ${target.type} ${target.clientTop}${target.clientLeft}${target.clientWidth}${target.clientHeight}${target.offsetTop}${target.offsetLeft}`;
};
-kpxcFields.prepareId = function(id) {
- return (id + '').replace(kpxcFields.rcssescape, kpxcFields.fcssescape);
+// Check for new password via autocomplete attribute
+kpxcFields.isAutocompleteAppropriate = function(field) {
+ const autocomplete = field.getLowerCaseAttribute('autocomplete');
+ return autocomplete !== 'new-password';
};
-/**
- * Returns the first parent element satifying the {@code predicate} mapped by {@code resultFn} or else {@code defaultVal}.
- * @param {HTMLElement} element The start element (excluded, starting with the parents)
- * @param {function} predicate Matcher for the element to find, type (HTMLElement) => boolean
- * @param {function} resultFn Callback function of type (HTMLElement) => {*} called for the first matching element
- * @param {fun} defaultValFn Fallback return value supplier, if no element matching the predicate can be found
- */
-kpxcFields.traverseParents = function(element, predicate, resultFn = () => true, defaultValFn = () => false) {
- for (let f = element.parentElement; f !== null; f = f.parentElement) {
- if (predicate(f)) {
- return resultFn(f);
- }
+// Checks if Custom Login Fields are used for the site
+kpxcFields.isCustomLoginFieldsUsed = function() {
+ const location = kpxc.getDocumentLocation();
+ return kpxc.settings['defined-custom-fields'] && kpxc.settings['defined-custom-fields'][location];
+};
+
+// Returns true if form is a search form
+kpxcFields.isSearchForm = function(form) {
+ // Check form action
+ const formAction = form.getLowerCaseAttribute('action');
+ if (formAction && (formAction.includes('search') && !formAction.includes('research'))) {
+ return true;
}
- return defaultValFn();
+ // Ignore form with search classes
+ const formId = form.getLowerCaseAttribute('id');
+ if (form.className && (form.className.includes('search')
+ || (formId && formId.includes('search') && !formId.includes('research')))) {
+ return true;
+ }
+
+ return false;
};
// Checks if input field is a search field. Attributes or form action containing 'search', or parent element holding
// role="search" will be identified as a search field.
kpxcFields.isSearchField = function(target) {
- const attributes = target.attributes;
-
// Check element attributes
- for (const attr of attributes) {
+ for (const attr of target.attributes) {
if ((attr.value && (attr.value.toLowerCase().includes('search')) || attr.value === 'q')) {
return true;
}
@@ -298,21 +441,8 @@ kpxcFields.isSearchField = function(target) {
// Check closest form
const closestForm = kpxc.getForm(target);
- if (closestForm) {
- // Check form action
- const formAction = closestForm.getAttribute('action');
- if (formAction && (formAction.toLowerCase().includes('search')
- && !formAction.toLowerCase().includes('research'))) {
- return true;
- }
-
- // Check form class and id
- const closestFormId = closestForm.getAttribute('id');
- const closestFormClass = closestForm.className;
- if (closestFormClass && (closestForm.className.toLowerCase().includes('search')
- || (closestFormId && closestFormId.toLowerCase().includes('search') && !closestFormId.toLowerCase().includes('research')))) {
- return true;
- }
+ if (closestForm && kpxcFields.isSearchForm(closestForm)) {
+ return true;
}
// Check parent elements for role='search'
@@ -323,9 +453,10 @@ kpxcFields.isSearchField = function(target) {
return false;
};
-kpxcFields.isVisible = function(field) {
+// Returns true if element is visible on the page
+kpxcFields.isVisible = function(elem) {
// Check element position and size
- const rect = field.getBoundingClientRect();
+ const rect = elem.getBoundingClientRect();
if (rect.x < 0
|| rect.y < 0
|| rect.width < 8
@@ -336,1703 +467,1251 @@ kpxcFields.isVisible = function(field) {
}
// Check CSS visibility
- const fieldStyle = getComputedStyle(field);
- if (fieldStyle.visibility && (fieldStyle.visibility === 'hidden' || fieldStyle.visibility === 'collapse')
- || fieldStyle.opacity === '0') {
+ const elemStyle = getComputedStyle(elem);
+ if (elemStyle.visibility && (elemStyle.visibility === 'hidden' || elemStyle.visibility === 'collapse')) {
return false;
}
// Check for parent opacity
- if (kpxcFields.traverseParents(field, f => f.style.opacity === '0')) {
+ if (kpxcFields.traverseParents(elem, f => f.style.opacity === '0')) {
return false;
}
return true;
};
-kpxcFields.isAutocompleteAppropriate = function(field) {
- const autocomplete = field.getLowerCaseAttribute('autocomplete');
- return autocomplete !== 'new-password';
-};
-
-kpxcFields.getAllFields = function() {
- const fields = [];
- const inputs = kpxcObserverHelper.getInputs(document);
-
- for (const i of inputs) {
- if (kpxcFields.isVisible(i) && !kpxcFields.isSearchField(i) && kpxcFields.isAutocompleteAppropriate(i)) {
- kpxcFields.setUniqueId(i);
- fields.push(i);
- }
- }
-
- _detectedFields = fields.length;
-
- // Show add username-only option for the site in popup
- if (!_singleInputEnabledForPage
- && fields.length === 1 && fields[0].getLowerCaseAttribute('type') !== 'password') {
- browser.runtime.sendMessage({
- action: 'username_field_detected',
- args: true
- });
- }
-
- return fields;
-};
-
-kpxcFields.prepareVisibleFieldsWithID = function(pattern) {
- const patterns = document.querySelectorAll(pattern);
- for (const i of patterns) {
- if (kpxcFields.isVisible(i)) {
- kpxcFields.setUniqueId(i);
- }
- }
+kpxcFields.prepareId = function(id) {
+ return (id + '').replace(kpxcFields.rcssescape, kpxcFields.fcssescape);
};
-kpxcFields.getAllCombinations = function(inputs) {
- const fields = [];
- let uField = null;
-
- for (const i of inputs) {
- if (i) {
- if (i.getLowerCaseAttribute('type') === 'password') {
- const uId = (!uField || uField.size < 1) ? null : uField.getAttribute('data-kpxc-id');
-
- const combination = {
- username: uId,
- password: i.getAttribute('data-kpxc-id')
- };
- fields.push(combination);
-
- // Reset selected username field
- uField = null;
- } else {
- // Username field
- uField = i;
- }
+/**
+ * Returns the first parent element satifying the {@code predicate} mapped by {@code resultFn} or else {@code defaultVal}.
+ * @param {HTMLElement} element The start element (excluded, starting with the parents)
+ * @param {function} predicate Matcher for the element to find, type (HTMLElement) => boolean
+ * @param {function} resultFn Callback function of type (HTMLElement) => {*} called for the first matching element
+ * @param {fun} defaultValFn Fallback return value supplier, if no element matching the predicate can be found
+ */
+kpxcFields.traverseParents = function(element, predicate, resultFn = () => true, defaultValFn = () => false) {
+ for (let f = element.parentElement; f !== null; f = f.parentElement) {
+ if (predicate(f)) {
+ return resultFn(f);
}
}
- if (_singleInputEnabledForPage && fields.length === 0 && uField) {
- const combination = {
- username: uField.getAttribute('data-kpxc-id'),
- password: null
- };
- fields.push(combination);
- }
-
- return fields;
+ return defaultValFn();
};
-kpxcFields.getCombination = async function(givenType, fieldId) {
- if (kpxcFields.combinations.length === 0) {
- if (kpxcFields.useDefinedCredentialFields()) {
- return kpxcFields.combinations[0];
- }
- }
- // Use defined credential fields (already loaded into combinations)
+// Use Custom Fields instead of detected combinations
+kpxcFields.useCustomLoginFields = async function() {
const location = kpxc.getDocumentLocation();
- if (kpxc.settings['defined-custom-fields'] && kpxc.settings['defined-custom-fields'][location]) {
- return kpxcFields.combinations[0];
+ const creds = kpxc.settings['defined-custom-fields'][location];
+ if (!creds.username && !creds.password && !creds.totp && creds.fields.length === 0) {
+ return;
}
- for (const c of kpxcFields.combinations) {
- if (c[givenType] === fieldId) {
- return c;
+ // Finds the input field based on the stored ID
+ const findInputField = async function(inputFields, id) {
+ if (id) {
+ const input = inputFields.find(e => kpxcFields.getId(e) === id);
+ if (input) {
+ return input;
+ }
}
- }
- // Find new combination
- let combination = {
- username: null,
- password: null
+ return null;
};
- let newCombi = false;
- if (givenType === 'username') {
- const passwordField = kpxcFields.getPasswordField(fieldId, true);
- let passwordId = null;
- if (passwordField) {
- passwordId = kpxcFields.prepareId(passwordField.getAttribute('data-kpxc-id'));
- }
- combination = {
- username: fieldId,
- password: passwordId
- };
- newCombi = true;
- } else if (givenType === 'password') {
- const usernameField = await kpxcFields.getUsernameField(fieldId, true);
- let usernameId = null;
- if (usernameField) {
- usernameId = kpxcFields.prepareId(usernameField.getAttribute('data-kpxc-id'));
+ // Get all input fields from the page without any extra filters
+ const inputFields = [];
+ document.querySelectorAll('input, select').forEach(e => {
+ if (e.type !== 'hidden' && !e.disabled) {
+ inputFields.push(e);
}
- combination = {
- username: usernameId,
- password: fieldId
- };
- newCombi = true;
- }
+ });
- if (combination.username || combination.password) {
- kpxcFields.combinations.push(combination);
- }
+ [ creds.username, creds.password, creds.totp ] = await Promise.all([
+ await findInputField(inputFields, creds.username),
+ await findInputField(inputFields, creds.password),
+ await findInputField(inputFields, creds.totp)
+ ]);
- if (combination.username) {
- if (kpxc.credentials.length > 0) {
- kpxc.preparePageForMultipleCredentials(kpxc.credentials);
+ // Handle StringFields
+ const stringFields = [];
+ for (const sf of creds.fields) {
+ const field = await findInputField(inputFields, sf);
+ if (field) {
+ stringFields.push(field);
}
}
- if (newCombi) {
- combination.isNew = true;
+ // Handle custom TOTP field
+ if (creds.totp) {
+ creds.totp.setAttribute('kpxc-defined', 'totp');
+ kpxcTOTPIcons.newIcon(creds.totp, kpxc.databaseState, true);
}
- return combination;
-};
-/**
-* Return the username field or null if it not exists
-*/
-kpxcFields.getUsernameField = async function(passwordId, checkDisabled) {
- const passwordField = _f(passwordId);
- if (!passwordField) {
- return null;
- }
-
- const form = kpxc.getForm(passwordField);
- let usernameField = null;
-
- // Search all inputs on this one form
- if (form) {
- const inputs = form.querySelectorAll(kpxcFields.inputQueryPattern);
- for (const i of inputs) {
- kpxcFields.setUniqueId(i);
- if (i.getAttribute('data-kpxc-id') === passwordId) {
- return false; // Break
- }
-
- if (i.getLowerCaseAttribute('type') === 'password') {
- return true; // Continue
- }
-
- if (kpxc.settings.showLoginFormIcon && await kpxc.passwordFilled() === false) {
- kpxcUsernameIcons.newIcon(usernameField);
- }
- usernameField = i;
- }
- } else {
- // Search all inputs on page
- const inputs = kpxcFields.getAllFields();
- kpxc.initPasswordGenerator(inputs);
- for (const i of inputs) {
- if (i.getAttribute('data-kpxc-id') === passwordId) {
- break;
- }
+ const combinations = [];
+ combinations.push({
+ username: creds.username,
+ password: creds.password,
+ passwordInputs: [ creds.password ],
+ totp: creds.totp,
+ fields: stringFields
+ });
- if (i.getLowerCaseAttribute('type') === 'password') {
- continue;
- }
+ return combinations;
+};
- usernameField = i;
+// Copied from Sizzle.js
+kpxcFields.rcssescape = /([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g;
+kpxcFields.fcssescape = function(ch, asCodePoint) {
+ if (asCodePoint) {
+ // U+0000 NULL becomes U+FFFD REPLACEMENT CHARACTER
+ if (ch === '\0') {
+ return '\uFFFD';
}
- }
- if (usernameField && !checkDisabled) {
- const usernameId = usernameField.getAttribute('data-kpxc-id');
- // Check if usernameField is already used by another combination
- for (const c of kpxcFields.combinations) {
- if (c.username === usernameId) {
- usernameField = null;
- break;
- }
- }
+ // Control characters and (dependent upon position) numbers get escaped as code points
+ return ch.slice(0, -1) + '\\' + ch.charCodeAt(ch.length - 1).toString(16) + ' ';
}
- kpxcFields.setUniqueId(usernameField);
- return usernameField;
+ // Other potentially-special ASCII characters get backslash-escaped
+ return '\\' + ch;
};
+
/**
-* Return the password field or null if it not exists
-*/
-kpxcFields.getPasswordField = function(usernameId, checkDisabled) {
- const usernameField = _f(usernameId);
- if (!usernameField) {
- return null;
- }
+ * @Object kpxc
+ * The main content script object.
+ */
+const kpxc = {};
+kpxc.combinations = [];
+kpxc.credentials = [];
+kpxc.databaseState = DatabaseState.DISCONNECTED;
+kpxc.detectedFields = 0;
+kpxc.inputs = [];
+kpxc.settings = {};
+kpxc.singleInputEnabledForPage = false;
+kpxc.submitUrl = null;
+kpxc.url = null;
- const form = kpxc.getForm(usernameField);
- let passwordField = null;
+// Add page to Site Preferences with Username-only detection enabled. Set from the popup
+kpxc.addToSitePreferences = async function(sites) {
+ kpxc.initSitePreferences();
- // Search all inputs on this one form
- if (form) {
- const inputs = form.querySelectorAll('input[type=\'password\']');
- if (inputs.length > 0) {
- passwordField = inputs[0];
- }
- if (passwordField && passwordField.length < 1) {
- passwordField = null;
- }
+ // Returns a predefined URL for certain sites
+ let site = kpxcSites.definedURL(trimURL(window.top.location.href));
- kpxcPasswordIcons.newIcon(kpxc.settings.usePasswordGeneratorIcons, passwordField, [], undefined, _databaseState);
- } else {
- // Search all inputs on page
- const inputs = kpxcFields.getAllFields();
- kpxc.initPasswordGenerator(inputs);
-
- let active = false;
- for (const i of inputs) {
- if (i.getAttribute('data-kpxc-id') === usernameId) {
- active = true;
- }
- if (active && i.getLowerCaseAttribute('type') === 'password') {
- passwordField = i;
- break;
- }
+ // Check if the site already exists -> update the current settings
+ let siteExists = false;
+ for (const existingSite of kpxc.settings['sitePreferences']) {
+ if (existingSite.url === site) {
+ existingSite.ignore = IGNORE_NOTHING;
+ existingSite.usernameOnly = true;
+ siteExists = true;
}
}
- if (passwordField && !checkDisabled) {
- const passwordId = passwordField.getAttribute('data-kpxc-id');
- // Check if passwordField is already used by another combination
- for (const c of kpxcFields.combinations) {
- if (c.password === passwordId) {
- passwordField = null;
- break;
- }
- }
+ if (!siteExists) {
+ // Add wildcard to the URL
+ site = site.slice(0, site.lastIndexOf('/') + 1) + '*';
+
+ kpxc.settings['sitePreferences'].push({
+ url: site,
+ ignore: IGNORE_NOTHING,
+ usernameOnly: true
+ });
}
- kpxcFields.setUniqueId(passwordField);
- return passwordField;
+ await sendMessage('save_settings', kpxc.settings);
+ sendMessage('username_field_detected', false);
};
-kpxcFields.prepareCombinations = async function(combinations) {
- if (combinations.length === 0) {
- return;
- }
+// Clears all from the content and background scripts, including autocomplete
+kpxc.clearAllFromPage = function() {
+ kpxc.credentials = [];
+ kpxc.inputs = [];
+ kpxcAutocomplete.elements = [];
+ _called.retrieveCredentials = false;
- // Only request this once if there are combinations available
- let passwordFilled;
- if (combinations.length > 0) {
- passwordFilled = await kpxc.passwordFilled();
+ if (kpxc.settings.autoCompleteUsernames) {
+ kpxcAutocomplete.closeList();
}
- for (const c of combinations) {
- const pwField = _f(c.password);
- // Needed for auto-complete: don't overwrite manually filled-in password field
- if (pwField && !pwField.getAttribute('kpxcFields-onChange')) {
- pwField.setAttribute('kpxcFields-onChange', true);
- pwField.addEventListener('change', function() {
- this.setAttribute('unchanged', false);
- });
- }
-
- const fieldId = c.password || c.username;
- const field = _f(fieldId);
-
- // If no username field is found, handle the single password field as such
- const usernameField = c.username ? _f(c.username) : field;
+ // Switch back to default popup
+ sendMessage('get_status', [ true ]); // This is an internal function call
+};
- if (kpxc.settings.showLoginFormIcon && passwordFilled === false) {
- kpxcUsernameIcons.newIcon(usernameField, _databaseState);
- }
+// Creates a new combination manually from active element
+kpxc.createCombination = async function(activeElement) {
+ const combination = {
+ username: null,
+ password: null,
+ passwordInputs: [],
+ form: activeElement.form
+ };
- // Initialize form-submit for remembering credentials
- if (field) {
- const form = kpxc.getForm(field);
- if (form && form.length > 0) {
- kpxcForm.init(form, c);
- }
- }
+ if (activeElement.getLowerCaseAttribute('type') === 'password') {
+ combination.password = activeElement;
+ } else {
+ combination.username = activeElement;
}
+
+ return combination;
};
-kpxcFields.useDefinedCredentialFields = function() {
- const location = kpxc.getDocumentLocation();
- if (kpxc.settings['defined-custom-fields'] && kpxc.settings['defined-custom-fields'][location]) {
- const creds = kpxc.settings['defined-custom-fields'][location];
-
- // Handle custom TOTP field
- if (_f(creds.totp)) {
- const totpField = _f(creds.totp);
- totpField.setAttribute('kpxc-defined', 'totp');
- kpxcTOTPIcons.newIcon(totpField, _databaseState, true);
- }
+// Switch credentials if database is changed or closed
+kpxc.detectDatabaseChange = async function(response) {
+ kpxc.databaseState = DatabaseState.LOCKED;
+ kpxc.clearAllFromPage();
+ kpxcIcons.switchIcons();
- let found = _f(creds.username) || _f(creds.password);
- for (const i of creds.fields) {
- if (_fs(i)) {
- found = true;
- break;
- }
- }
+ if (document.visibilityState !== 'hidden') {
+ if (response.hash.new !== '' && response.hash.new !== response.hash.old) {
+ _called.retrieveCredentials = false;
+ const settings = await sendMessage('load_settings');
+ kpxc.settings = settings;
+ kpxc.databaseState = DatabaseState.UNLOCKED;
- if (found) {
- if (creds.username) {
- _f(creds.username).setAttribute('kpxc-defined', 'username');
- }
+ await kpxc.initCredentialFields();
+ kpxcIcons.switchIcons();
- if (creds.password) {
- _f(creds.password).setAttribute('kpxc-defined', 'password');
+ // If user has requested a manual fill through context menu the actual credential filling
+ // is handled here when the opened database has been regognized. It's not a pretty hack.
+ const manualFill = await sendMessage('page_get_manual_fill');
+ if (manualFill !== ManualFill.NONE) {
+ await kpxc.fillInFromActiveElement(manualFill === ManualFill.PASSWORD);
+ await sendMessage('page_set_manual_fill', ManualFill.NONE);
}
-
- const fields = {
- username: creds.username,
- password: creds.password,
- fields: creds.fields
- };
- kpxcFields.combinations = [];
- kpxcFields.combinations.push(fields);
-
- return true;
+ } else if (!response.connected) {
+ kpxc.databaseState = DatabaseState.DISCONNECTED;
+ kpxcIcons.switchIcons();
}
}
-
- return false;
};
-const kpxcObserverHelper = {};
-
-kpxcObserverHelper.inputTypes = [
- 'text',
- 'email',
- 'password',
- 'tel',
- 'number',
- 'username', // Note: Not a standard
- undefined, // Input field can be without any type. Include this and null to the list.
- null
-];
-
-// Ignores all nodes that doesn't contain elements
-// Also ignore few Youtube-specific custom nodeNames
-kpxcObserverHelper.ignoredNode = function(target) {
- if (target.nodeType === Node.ATTRIBUTE_NODE
- || target.nodeType === Node.TEXT_NODE
- || target.nodeType === Node.CDATA_SECTION_NODE
- || target.nodeType === Node.PROCESSING_INSTRUCTION_NODE
- || target.nodeType === Node.COMMENT_NODE
- || target.nodeType === Node.DOCUMENT_TYPE_NODE
- || target.nodeType === Node.NOTATION_NODE
- || target.nodeName === 'HTML'
- || target.nodeName === 'LINK'
- || target.nodeName === 'HEAD'
- || target.nodeName === 'VIDEO'
- || target.nodeName.startsWith('YTMUSIC')
- || target.nodeName.startsWith('YT-')) {
- return true;
- }
-
- return false;
-};
-
-kpxcObserverHelper.getInputs = function(target, ignoreVisibility = false) {
- // Ignores target element if it's not an element node
- if (kpxcObserverHelper.ignoredNode(target)) {
- return [];
+// Fill requested from the context menu. Active element is used for combination detection
+kpxc.fillInFromActiveElement = async function(passOnly = false) {
+ const el = document.activeElement;
+ if (el.nodeName !== 'INPUT' || kpxc.credentials.length === 0) {
+ return;
+ } else if (kpxc.credentials.length > 1 && kpxc.combinations.length > 0) {
+ kpxcAutocomplete.showList(el);
+ return;
}
- // Filter out any input fields with type 'hidden' right away
- const inputFields = [];
- Array.from(target.getElementsByTagName('input')).forEach(e => {
- if (e.type !== 'hidden'
- && !e.disabled
- && !kpxcObserverHelper.hasKpxcClass(e)) {
- inputFields.push(e);
- }
- });
-
- if (inputFields.length === 0) {
- return [];
+ // No previous combinations detected. Create a new one from active element
+ let combination;
+ if (kpxc.combinations.length === 0) {
+ combination = await kpxc.createCombination(el);
+ } else {
+ combination = el.type === 'password'
+ ? await kpxcFields.getCombination(el, 'password')
+ : await kpxcFields.getCombination(el, 'username');
}
- // Do not allow more visible inputs than _maximumInputs (default value: 100) -> return the first 100
- if (inputFields.length > _maximumInputs) {
- return inputFields.slice(0, _maximumInputs);
+ // Do not allow filling password to a non-password field
+ if (passOnly && combination && !combination.password) {
+ kpxcUI.createNotification('warning', tr('fieldsNoPasswordField'));
+ return;
}
- // Only include input fields that match with kpxcObserverHelper.inputTypes
- const inputs = [];
- for (const field of inputFields) {
- const type = field.getLowerCaseAttribute('type');
-
- if (ignoreVisibility) {
- if (kpxcObserverHelper.inputTypes.includes(type)) {
- inputs.push(field);
- }
- } else {
- if (kpxcObserverHelper.inputTypes.includes(type) && kpxcFields.isVisible(field)) {
- kpxcFields.setUniqueId(field);
- inputs.push(field);
- }
- }
- }
- return inputs;
+ await sendMessage('page_set_login_id', 0);
+ kpxc.fillInCredentials(combination, kpxc.credentials[0].login, kpxc.credentials[0].uuid, passOnly);
};
-// Gets of generates an ID for the element
-kpxcObserverHelper.getId = function(target) {
- if (target.classList.length > 0) {
- return target.classlist;
+// Fill requested by Auto-Fill
+kpxc.fillFromAutofill = async function() {
+ if (kpxc.credentials.length !== 1 || kpxc.combinations.length !== 1) {
+ return;
}
- if (target.id !== '') {
- return target.id;
- }
+ await sendMessage('page_set_login_id', 0);
+ kpxc.fillInCredentials(kpxc.combinations[0], kpxc.credentials[0].login, kpxc.credentials[0].uuid);
- return `kpxc${target.clientTop}${target.clientLeft}${target.clientWidth}${target.clientHeight}`;
+ // Generate popup-list of usernames + descriptions
+ sendMessage('popup_login', [ `${kpxc.credentials[0].login} (${kpxc.credentials[0].name})` ]);
};
-kpxcObserverHelper.hasKpxcClass = function(target) {
- if (!target.className
- || !target.className.includes('kpxc')) {
- return false;
+// Fill requested by selecting credentials from the popup
+kpxc.fillFromPopup = async function(id, uuid) {
+ if (!kpxc.credentials.length === 0 || !kpxc.credentials[id] || kpxc.combinations.length === 0) {
+ return;
}
- return target.className.includes('kpxc');
+ await sendMessage('page_set_login_id', id);
+ kpxc.fillInCredentials(kpxc.combinations[0], kpxc.credentials[id].login, uuid);
+ kpxcAutocomplete.closeList();
};
-kpxcObserverHelper.ignoredElement = function(target) {
- if (kpxcObserverHelper.ignoredNode(target)) {
- return true;
- }
+// Fill requested from TOTP icon
+kpxc.fillFromTOTP = async function(target) {
+ const el = target || document.activeElement;
+ let index = await sendMessage('page_get_login_id');
- // Ignore elements that do not have a className (including SVG)
- if (typeof target.className !== 'string') {
- return true;
+ // Use the first credential available if not set
+ if (index === undefined) {
+ index = 0;
}
- // Ignore KeePassXC-Browser classes
- if (kpxcObserverHelper.hasKpxcClass(target)) {
- return true;
- }
+ if (index >= 0 && kpxc.credentials[index]) {
+ // Check the value from StringFields
+ if (kpxc.credentials[index].stringFields && kpxc.credentials[index].stringFields.length > 0) {
+ const stringFields = kpxc.credentials[index].stringFields;
+ for (const s of stringFields) {
+ const val = s['KPH: {TOTP}'];
+ if (val) {
+ kpxc.setValue(el, val);
+ }
+ }
+ } else if (kpxc.credentials[index].totp && kpxc.credentials[index].totp.length > 0) {
+ // Retrieve a new TOTP value
+ const totp = await sendMessage('get_totp', [ kpxc.credentials[index].uuid, kpxc.credentials[index].totp ]);
+ if (!totp) {
+ kpxcUI.createNotification('warning', tr('credentialsNoTOTPFound'));
+ return;
+ }
- return false;
+ kpxc.setValue(el, totp);
+ }
+ }
};
-kpxcObserverHelper.handleObserverAdd = function(target) {
- if (kpxcObserverHelper.ignoredElement(target)) {
+// Fill requested from username icon
+kpxc.fillFromUsernameIcon = async function(combination) {
+ if (kpxc.credentials.length === 0) {
return;
- }
-
- const inputs = kpxcObserverHelper.getInputs(target);
- if (inputs.length === 0) {
+ } else if (kpxc.credentials.length > 1) {
+ kpxcAutocomplete.showList(combination.username || combination.password);
return;
}
- const neededLength = _detectedFields === 1 ? 0 : 1;
- const id = kpxcObserverHelper.getId(target);
- if (inputs.length > neededLength && !_observerIds.includes(id)) {
- // Save target element id for preventing multiple calls to initCredentialsFields()
- _observerIds.push(id);
-
- // Sometimes the settings haven't been loaded before new input fields are detected
- if (Object.keys(kpxc.settings).length === 0) {
- kpxc.init();
- } else {
- kpxc.handleCredentialFields(inputs);
- }
- }
+ await sendMessage('page_set_login_id', 0);
+ kpxc.fillInCredentials(combination, kpxc.credentials[0].login, kpxc.credentials[0].uuid);
};
-kpxcObserverHelper.handleObserverRemove = function(target) {
- if (kpxcObserverHelper.ignoredElement(target)) {
+/**
+ * The main function for filling any credentials
+ * @param {Array} combination Combination to be used
+ * @param {String} predefinedUsername Predefined username. If set, there's no need to find it from combinations
+ * @param {Boolean} passOnly If only password is filled
+ * @param {String} uuid Identifier for the entry. There can be identical usernames with different password
+ */
+kpxc.fillInCredentials = async function(combination, predefinedUsername, uuid, passOnly = false) {
+ if (kpxc.credentials.length === 0) {
+ kpxcUI.createNotification('error', tr('credentialsNoLoginsFound'));
return;
}
- const inputs = kpxcObserverHelper.getInputs(target, true);
- if (inputs.length === 0) {
+ if (!combination || (!combination.username && !combination.password)) {
return;
}
- kpxc.deleteHiddenIcons();
-
- // Remove target element id from the list
- const id = kpxcObserverHelper.getId(target);
- if (_observerIds.includes(id)) {
- const index = _observerIds.indexOf(id);
- if (index >= 0) {
- _observerIds.splice(index, 1);
- }
+ // Use predefined username as default
+ let usernameValue = predefinedUsername;
+ if (!usernameValue) {
+ // With single password field the combination.password is used instead
+ usernameValue = combination.username ? combination.username.value : combination.password.value;
}
-};
-kpxcObserverHelper.detectURLChange = function() {
- if (_documentURL !== document.location.href) {
- _documentURL = document.location.href;
- kpxcEvents.clearCredentials();
- kpxc.initCredentialFields(true);
+ // Find the correct credentials
+ const selectedCredentials = kpxc.credentials.find(c => c.uuid === uuid);
+ if (!selectedCredentials) {
+ return;
}
-};
-MutationObserver = window.MutationObserver || window.WebKitMutationObserver;
-
-
-const kpxc = {};
-kpxc.settings = {};
-kpxc.u = null;
-kpxc.p = null;
-kpxc.url = null;
-kpxc.submitUrl = null;
-kpxc.credentials = [];
-kpxc.observer = null;
-
-const initContentScript = async function() {
- try {
- const settings = await browser.runtime.sendMessage({
- action: 'load_settings'
- });
-
- kpxc.settings = settings;
+ // Handle auto-submit
+ let skipAutoSubmit = false;
+ if (selectedCredentials.skipAutoSubmit !== undefined) {
+ skipAutoSubmit = selectedCredentials.skipAutoSubmit === 'true';
+ }
- // Don't initialize MutationObserver if the site is ignored
- if (kpxc.siteIgnored()) {
- return;
- }
+ // Fill password
+ if (combination.password) {
+ kpxc.setValueWithChange(combination.password, selectedCredentials.password);
+ await kpxc.setPasswordFilled(true);
+ }
- if (kpxc.settings.useObserver && !kpxc.observer) {
- kpxc.initObserver();
+ // Fill username
+ if (combination.username && (!combination.username.value || combination.username.value !== usernameValue)) {
+ if (!passOnly) {
+ kpxc.setValueWithChange(combination.username, usernameValue);
}
+ }
- await kpxc.updateDatabaseState();
- kpxc.switchIcons();
- await kpxc.initCredentialFields();
-
- // Retrieve submitted credentials if available.
- const creds = await browser.runtime.sendMessage({
- action: 'page_get_submitted'
- });
-
- const redirectCount = await browser.runtime.sendMessage({
- action: 'page_get_redirect_count'
- });
+ // Fill StringFields
+ if (selectedCredentials.stringFields.length > 0) {
+ kpxc.fillInStringFields(combination.fields, selectedCredentials.stringFields);
+ }
- if (creds && creds.submitted) {
- // If username field is not set, wait for credentials in kpxc.retrieveCredentialsCallback.
- if (!creds.username) {
- return;
- }
+ // Close autocomplete menu after fill
+ kpxcAutocomplete.closeList();
- if (redirectCount >= kpxc.settings.redirectAllowance) {
- await browser.runtime.sendMessage({
- action: 'page_clear_submitted'
- });
- }
+ // Reset ManualFill
+ await sendMessage('page_set_manual_fill', ManualFill.NONE);
- kpxc.rememberCredentials(creds.username, creds.password, creds.url, creds.oldCredentials);
+ // Auto-submit
+ if (kpxc.settings.autoSubmit && !skipAutoSubmit) {
+ const submitButton = kpxc.getFormSubmitButton(combination.form);
+ if (submitButton !== undefined) {
+ submitButton.click();
+ } else {
+ combination.form.submit();
}
- } catch (err) {
- console.log('initContentScript error: ', err);
}
};
-if (document.readyState === 'complete' || (document.readyState !== 'loading' && !document.documentElement.doScroll)) {
- initContentScript();
-} else {
- document.addEventListener('DOMContentLoaded', initContentScript);
-}
-
-kpxc.init = function() {
- initContentScript();
-};
-
-// Detects DOM changes in the document
-kpxc.initObserver = function() {
- kpxc.observer = new MutationObserver(function(mutations, obs) {
- if (document.visibilityState === 'hidden' || kpxcUI.mouseDown) {
- return;
- }
-
- // Limit the mutation handling
- if (mutations.length > _maximumMutations) {
- mutations.slice(0, _maximumMutations);
- }
-
- for (const mut of mutations) {
- // Skip text nodes and base HTML element
- if (kpxcObserverHelper.ignoredNode(mut.target)) {
- continue;
- }
-
- // Check document URL change and detect new fields
- kpxcObserverHelper.detectURLChange();
-
- // Handle attributes only if CSS display is modified
- if (mut.type === 'attributes') {
- // Check if some class is changed that holds a form or input field(s). Ignore large forms.
- const formInput = mut.target.querySelector('form input');
- if (mut.attributeName === 'class' && formInput !== null && formInput.form.length < 20) {
- kpxc.handleCredentialFields(kpxcObserverHelper.getInputs(formInput.form));
- continue;
- }
+// Fills StringFields defined in Custom Fields
+kpxc.fillInStringFields = function(fields, stringFields) {
+ const filledInFields = [];
+ if (fields && stringFields && fields.length > 0 && stringFields.length > 0) {
+ for (let i = 0; i < fields.length; i++) {
+ const currentField = fields[i];
+ const stringFieldValue = Object.values(stringFields[i]);
- const newValue = mut.target.getAttribute(mut.attributeName);
- if (newValue && (newValue.includes('display') || newValue.includes('z-index'))) {
- if (mut.target.style.display !== 'none') {
- kpxcObserverHelper.handleObserverAdd(mut.target);
- } else {
- kpxcObserverHelper.handleObserverRemove(mut.target);
- }
- }
- } else if (mut.type === 'childList') {
- kpxcObserverHelper.handleObserverAdd((mut.addedNodes.length > 0) ? mut.addedNodes[0] : mut.target);
- kpxcObserverHelper.handleObserverRemove((mut.removedNodes.length > 0) ? mut.removedNodes[0] : mut.target);
+ if (currentField && stringFieldValue[0]) {
+ kpxc.setValue(currentField, stringFieldValue[0]);
+ filledInFields.push(currentField);
}
}
- });
-
- // define what element should be observed by the observer
- // and what types of mutations trigger the callback
- kpxc.observer.observe(document, {
- subtree: true,
- attributes: true,
- childList: true,
- characterData: true,
- attributeFilter: [ 'style', 'class' ]
- });
+ }
};
-// Clears all from the content and background scripts, including autocomplete
-kpxc.clearAllFromPage = function() {
- kpxcEvents.clearCredentials();
- browser.runtime.sendMessage({
- action: 'page_clear_logins'
- });
-
- // Switch back to default popup
- browser.runtime.sendMessage({
- action: 'get_status',
- args: [ true ] // Set polling to true, this is an internal function call
- });
+// Get location URL by domain or full URL
+kpxc.getDocumentLocation = function() {
+ return kpxc.settings.saveDomainOnly ? document.location.origin : document.location.href;
};
-// Switch credentials if database is changed or closed
-kpxc.detectDatabaseChange = async function(response) {
- _databaseState = DatabaseState.LOCKED;
- kpxc.clearAllFromPage();
- kpxc.switchIcons();
-
- if (document.visibilityState !== 'hidden') {
- if (response.hash.new !== '' && response.hash.new !== response.hash.old) {
- _called.retrieveCredentials = false;
- const settings = await browser.runtime.sendMessage({
- action: 'load_settings'
- });
- kpxc.settings = settings;
- await kpxc.initCredentialFields(true);
- _databaseState = DatabaseState.UNLOCKED;
- kpxc.switchIcons();
+// Returns the form that includes the inputField
+kpxc.getForm = function(inputField) {
+ if (inputField.form) {
+ return inputField.form;
+ }
- // If user has requested a manual fill through context menu the actual credential filling
- // is handled here when the opened database has been regognized. It's not a pretty hack.
- if (_called.manualFillRequested && _called.manualFillRequested !== ManualFill.NONE) {
- await kpxc.fillInFromActiveElement(false, _called.manualFillRequested === ManualFill.PASS);
- _called.manualFillRequested = ManualFill.NONE;
+ for (const f of document.forms) {
+ for (const e of f.elements) {
+ if (e === inputField) {
+ return f;
}
- } else if (!response.connected) {
- _databaseState = DatabaseState.DISCONNECTED;
- kpxc.switchIcons();
}
}
};
-// Checks if the site has been ignored using Site Preferences
-kpxc.siteIgnored = function(condition) {
- kpxc.initializeSitePreferences();
- if (kpxc.settings.sitePreferences) {
- let currentLocation;
- try {
- currentLocation = window.top.location.href;
- } catch (err) {
- // Cross-domain security error inspecting window.top.location.href.
- // This catches an error when an iframe is being accessed from another (sub)domain -> use the iframe URL instead.
- currentLocation = window.self.location.href;
- }
+// Returns form action URL or document origin if it's not found
+kpxc.getFormActionUrl = function(combination) {
+ if (!combination || (combination.username === null && combination.password === null)) {
+ return null;
+ }
- const currentSetting = condition || IGNORE_FULL;
- for (const site of kpxc.settings.sitePreferences) {
- if (siteMatch(site.url, currentLocation) || site.url === currentLocation) {
- if (site.ignore === currentSetting) {
- return true;
- }
+ let action = null;
+ if (combination.form && combination.form.length > 0) {
+ action = combination.form.action;
+ }
- _singleInputEnabledForPage = site.usernameOnly;
- }
- }
+ if (typeof(action) !== 'string' || action === '') {
+ action = document.location.origin + document.location.pathname;
}
- return false;
+ return action;
};
-// Initializes all input fields from the whole page
-kpxc.initCredentialFields = async function(forceCall) {
- if (_called.initCredentialFields && !forceCall) {
- return;
+// Get site URL in a proper form
+kpxc.getSite = function(sites) {
+ if (!sites || sites.length === 0) {
+ return '';
}
- _called.initCredentialFields = true;
- await browser.runtime.sendMessage({
- action: 'page_clear_logins',
- args: _called.clearLogins
- });
-
- _called.clearLogins = true;
+ let site = trimURL(sites[0]);
+ kpxc.initSitePreferences();
- if (kpxc.siteIgnored()) {
- return;
+ if (slashNeededForUrl(site)) {
+ site += '/';
}
- // Get inputs from the whole document
- const inputs = kpxcFields.getAllFields();
- kpxc.handleCredentialFields(inputs);
+ return site;
};
-// Handles the input fields from the whole page or from dynamically added content
-kpxc.handleCredentialFields = async function(inputs) {
- if (inputs.length === 0) {
- return;
- }
+// Identifies all forms in the page
+kpxc.identifyFormInputs = async function() {
+ const forms = [];
+ const documentForms = document.forms; // Cache the value just in case
- if (!kpxcFields.useDefinedCredentialFields()) {
- // Get all combinations of username + password fields
- kpxcFields.combinations = kpxcFields.getAllCombinations(inputs);
- }
- kpxcFields.prepareCombinations(kpxcFields.combinations);
+ for (const form of documentForms) {
+ if (!kpxcFields.isVisible(form)) {
+ continue;
+ }
- if (kpxc.settings.showOTPIcon) {
- kpxc.initOTPFields(inputs);
+ if (kpxcFields.isSearchForm(form)) {
+ continue;
+ }
+
+ forms.push(form);
}
- if (kpxcFields.combinations.length === 0 && inputs.length === 0) {
- browser.runtime.sendMessage({
- action: 'show_default_browseraction'
- });
- return;
+ // Identify input fields in the saved forms
+ const inputs = [];
+ for (const form of forms) {
+ const formInputs = kpxcObserverHelper.getInputs(form);
+ for (const f of formInputs) {
+ inputs.push(f);
+ }
}
- kpxc.url = document.location.href;
- kpxc.submitUrl = kpxc.getFormActionUrl(kpxcFields.combinations[0]);
+ await kpxc.initCombinations(inputs);
+ return inputs;
+};
+
+// Ignore a certain site
+kpxc.ignoreSite = async function(sites) {
+ const site = kpxc.getSite(sites);
- // Get submitUrl for a single input
- if (!kpxc.submitUrl && kpxcFields.combinations.length === 1 && inputs.length === 1) {
- kpxc.submitUrl = kpxc.getFormActionUrlFromSingleInput(inputs[0]);
+ // Check if the site already exists
+ let siteExists = false;
+ for (const existingSite of kpxc.settings['sitePreferences']) {
+ if (existingSite.url === site) {
+ existingSite.ignore = IGNORE_NORMAL;
+ siteExists = true;
+ }
}
- if (kpxc.settings.autoRetrieveCredentials && _called.retrieveCredentials === false && (kpxc.url && kpxc.submitUrl)) {
- _called.retrieveCredentials = true;
- kpxc.retrieveCredentialsCallback(await browser.runtime.sendMessage({
- action: 'retrieve_credentials',
- args: [ kpxc.url, kpxc.submitUrl ]
- }));
- } else if (_singleInputEnabledForPage) {
- kpxc.preparePageForMultipleCredentials(kpxc.credentials);
+ if (!siteExists) {
+ kpxc.settings['sitePreferences'].push({
+ url: site,
+ ignore: IGNORE_NORMAL,
+ usernameOnly: false
+ });
}
+
+ await sendMessage('save_settings', kpxc.settings);
};
-kpxc.initPasswordGenerator = function(inputs) {
- if (!kpxc.settings.usePasswordGeneratorIcons) {
+// Initialize autocomplete for username fields
+kpxc.initAutocomplete = function() {
+ if (!kpxc.settings.autoCompleteUsernames) {
return;
}
- for (let i = 0; i < inputs.length; i++) {
- if (inputs[i] && inputs[i].getLowerCaseAttribute('type') === 'password') {
- kpxcPasswordIcons.newIcon(true, inputs[i], inputs, i, _databaseState);
+ for (const c of kpxc.combinations) {
+ if (c.username) {
+ kpxcAutocomplete.create(c.username, false, kpxc.settings.autoSubmit);
+ } else if (!c.username && c.password) {
+ // Single password field
+ kpxcAutocomplete.create(c.password, false, kpxc.settings.autoSubmit);
}
}
};
-kpxc.initOTPFields = function(inputs) {
- for (const i of inputs) {
- const id = i.getLowerCaseAttribute('id');
- const name = i.getLowerCaseAttribute('name');
- const autocomplete = i.getLowerCaseAttribute('autocomplete');
+// Looks for any username & password combinations from the detected input fields
+kpxc.initCombinations = async function(inputs = []) {
+ if (inputs.length === 0) {
+ return [];
+ }
- if (autocomplete === 'one-time-code' || acceptedOTPFields.some(f => (id && id.includes(f)) || (name && name.includes(f)))) {
- kpxcTOTPIcons.newIcon(i, _databaseState);
- }
+ const combinations = kpxcFields.isCustomLoginFieldsUsed()
+ ? await kpxcFields.useCustomLoginFields()
+ : await kpxcFields.getAllCombinations(inputs);
+ if (!combinations || combinations.length === 0) {
+ return [];
}
-};
-kpxc.receiveCredentialsIfNecessary = async function() {
- if (kpxc.credentials.length === 0 && _called.retrieveCredentials === false) {
- const credentials = await browser.runtime.sendMessage({
- action: 'retrieve_credentials',
- args: [ kpxc.url, kpxc.submitUrl, true ] // Sets triggerUnlock to true
- });
+ for (const c of combinations) {
+ // If no username field is found, handle the single password field as such
+ const field = c.username || c.password;
+ if (field && c.form) {
+ // Initialize form-submit for remembering credentials
+ kpxcForm.init(c.form, c);
+ }
- // If the database was locked, this is scope never met. In these cases the response is met at kpxc.detectDatabaseChange
- _called.manualFillRequested = ManualFill.NONE;
- await kpxc.retrieveCredentialsCallback(credentials, false);
- return credentials;
+ // Don't allow duplicates
+ if (!kpxc.combinations.some(f => f.username === c.username && f.password === c.password && f.totp === c.totp && f.form === c.form)) {
+ kpxc.combinations.push(c);
+ }
}
- return kpxc.credentials;
+ return combinations;
};
-kpxc.retrieveCredentialsCallback = async function(credentials, dontAutoFillIn) {
- if (kpxcFields.combinations.length > 0) {
- kpxc.u = _f(kpxcFields.combinations[0].username);
- kpxc.p = _f(kpxcFields.combinations[0].password);
- }
-
- if (credentials && credentials.length > 0) {
- kpxc.credentials = credentials;
- await kpxc.prepareFieldsForCredentials(!dontAutoFillIn);
- }
+// The main function for finding input fields
+kpxc.initCredentialFields = async function() {
+ await sendMessage('page_clear_logins', _called.clearLogins );
+ _called.clearLogins = true;
- // Retrieve submitted credentials if available
- const creds = await browser.runtime.sendMessage({
- action: 'page_get_submitted'
- });
+ // Identify all forms in the page
+ const formInputs = await kpxc.identifyFormInputs();
- if (creds && creds.submitted) {
- await browser.runtime.sendMessage({
- action: 'page_clear_submitted'
- });
+ // Search all remaining inputs from the page, ignore the previous input fields
+ const pageInputs = await kpxcFields.getAllPageInputs(formInputs);
+ if (formInputs.length === 0 && pageInputs.length === 0) {
+ // Run 'redetect_credentials' manually if no fields are found after a page load
+ setTimeout(async function() {
+ if (kpxc.inputs.length === 0 || kpxc.combinations.length === 0) {
+ kpxc.initCredentialFields();
+ }
+ }, 2000);
- kpxc.rememberCredentials(creds.username, creds.password, creds.url, creds.oldCredentials);
+ return;
}
-};
-kpxc.prepareFieldsForCredentials = async function(autoFillInForSingle) {
- // Only one login for this site
- if (autoFillInForSingle && kpxc.settings.autoFillSingleEntry && kpxc.credentials.length === 1) {
- let combination = null;
- if (!kpxc.p && !kpxc.u && kpxcFields.combinations.length > 0) {
- kpxc.u = _f(kpxcFields.combinations[0].username);
- kpxc.p = _f(kpxcFields.combinations[0].password);
- combination = kpxcFields.combinations[0];
- }
- if (kpxc.u) {
- kpxc.setValueWithChange(kpxc.u, kpxc.credentials[0].login);
- combination = await kpxcFields.getCombination('username', kpxc.u);
- }
- if (kpxc.p) {
- kpxc.setValueWithChange(kpxc.p, kpxc.credentials[0].password);
- combination = await kpxcFields.getCombination('password', kpxc.p);
- }
+ // Combine inputs
+ kpxc.inputs = [...formInputs, ...pageInputs];
- if (combination) {
- const list = [];
- if (kpxc.fillInStringFields(combination.fields, kpxc.credentials[0].stringFields, list)) {
- kpxcForm.destroy(false, { 'password': list.list[0], 'username': list.list[1] });
- }
- }
+ // Combinations are already saved when identifying fields
+ if (kpxc.combinations.length === 0) {
+ sendMessage('show_default_browseraction');
+ return;
+ }
- // Generate popup-list of usernames + descriptions
- browser.runtime.sendMessage({
- action: 'popup_login',
- args: [ `${kpxc.credentials[0].login} (${kpxc.credentials[0].name})` ]
- });
- } else if (kpxc.credentials.length > 1 || (kpxc.credentials.length > 0 && (!kpxc.settings.autoFillSingleEntry || !autoFillInForSingle))) {
- kpxc.preparePageForMultipleCredentials(kpxc.credentials);
+ await kpxcIcons.initIcons(kpxc.combinations);
+
+ if (kpxc.databaseState === DatabaseState.UNLOCKED) {
+ await kpxc.retrieveCredentials();
}
};
-kpxc.preparePageForMultipleCredentials = function(credentials) {
- if (credentials.length === 0) {
+// Intializes the login popup list for choosing credentials
+kpxc.initLoginPopup = function() {
+ if (kpxc.credentials.length === 0) {
return;
}
- function getLoginText(credential, withGroup) {
+ const getLoginText = function(credential, withGroup) {
const group = (withGroup && credential.group) ? `[${credential.group}] ` : '';
const visibleLogin = (credential.login.length > 0) ? credential.login : tr('credentialsNoUsername');
const text = `${group}${credential.name} (${visibleLogin})`;
+
if (credential.expired && credential.expired === 'true') {
return `${text} [${tr('credentialExpired')}]`;
}
+
return text;
- }
+ };
- function getUniqueGroupCount(creds) {
+ const getUniqueGroupCount = function(creds) {
const groups = creds.map(c => c.group || '');
const uniqueGroups = new Set(groups);
return uniqueGroups.size;
- }
+ };
// Add usernames + descriptions to autocomplete-list and popup-list
const usernames = [];
kpxcAutocomplete.elements = [];
- const showGroupNameInAutocomplete = kpxc.settings.showGroupNameInAutocomplete && (getUniqueGroupCount(credentials) > 1);
- for (let i = 0; i < credentials.length; i++) {
- const loginText = getLoginText(credentials[i], showGroupNameInAutocomplete);
- usernames.push(loginText);
+ const showGroupNameInAutocomplete = kpxc.settings.showGroupNameInAutocomplete && (getUniqueGroupCount(kpxc.credentials) > 1);
+
+ for (let i = 0; i < kpxc.credentials.length; i++) {
+ const loginText = getLoginText(kpxc.credentials[i], showGroupNameInAutocomplete);
+ usernames.push({ text: loginText, uuid: kpxc.credentials[i].uuid });
- const item = {
+ kpxcAutocomplete.elements.push({
label: loginText,
- value: credentials[i].login,
+ value: kpxc.credentials[i].login,
+ uuid: kpxc.credentials[i].uuid,
loginId: i
- };
- kpxcAutocomplete.elements.push(item);
+ });
}
// Generate popup-list of usernames + descriptions
- browser.runtime.sendMessage({
- action: 'popup_login',
- args: usernames
- });
-
- // Initialize autocomplete for username fields
- if (kpxc.settings.autoCompleteUsernames) {
- for (const i of kpxcFields.combinations) {
- // Both username and password fields are visible
- if (_detectedFields >= 2) {
- if (_f(i.username)) {
- kpxcAutocomplete.create(_f(i.username), false, kpxc.settings.autoSubmit);
- }
- } else if (_detectedFields === 1) {
- if (_f(i.username)) {
- kpxcAutocomplete.create(_f(i.username), false, kpxc.settings.autoSubmit);
- }
- if (_f(i.password)) {
- kpxcAutocomplete.create(_f(i.password), false, kpxc.settings.autoSubmit);
- }
- }
- }
- }
+ sendMessage('popup_login', usernames);
};
-// Returns the form that includes the inputField
-kpxc.getForm = function(inputField) {
- if (inputField.form) {
- return inputField.form;
+// Delete previously created Object if it exists. It will be replaced by an Array
+kpxc.initSitePreferences = function() {
+ if (kpxc.settings['sitePreferences'] !== undefined && kpxc.settings['sitePreferences'].constructor === Object) {
+ delete kpxc.settings['sitePreferences'];
}
- for (const f of document.forms) {
- for (const e of f.elements) {
- if (e === inputField) {
- return f;
- }
- }
+ if (!kpxc.settings['sitePreferences']) {
+ kpxc.settings['sitePreferences'] = [];
}
};
-kpxc.getFormActionUrl = function(combination) {
- if (!combination) {
- return null;
- }
-
- const field = _f(combination.password) || _f(combination.username);
- if (field === null) {
- return null;
- }
-
- const form = kpxc.getForm(field);
- let action = null;
+kpxc.passwordFilled = async function() {
+ return await sendMessage('password_get_filled');
+};
- if (form && form.length > 0) {
- action = form.action;
+// Prepares autocomplete and login popup ready for user interaction
+kpxc.prepareCredentials = async function() {
+ if (kpxc.credentials.length === 0) {
+ return;
}
- if (typeof(action) !== 'string' || action === '') {
- action = document.location.origin + document.location.pathname;
+ if (kpxc.settings.autoFillSingleEntry && kpxc.credentials.length === 1) {
+ kpxc.fillFromAutofill();
+ return;
}
- return action;
+ kpxc.initLoginPopup();
+ kpxc.initAutocomplete();
};
-kpxc.getFormActionUrlFromSingleInput = function(field) {
- if (!field) {
- return null;
- }
-
- let action = field.formAction;
-
- if (typeof(action) !== 'string' || action === '') {
- action = document.location.origin + document.location.pathname;
+/**
+ * Gets the credential list and shows the update banner
+ * @param {string} usernameValue Submitted username
+ * @param {string} passwordValue Submitted password
+ * @param {string} urlValue URL of the page where password change was detected
+ * @param {Array} oldCredentials Credentials saved from the password change page, if available
+ */
+kpxc.rememberCredentials = async function(usernameValue, passwordValue, urlValue, oldCredentials) {
+ const credentials = (oldCredentials !== undefined && oldCredentials.length > 0) ? oldCredentials : kpxc.credentials;
+ if (passwordValue === '') {
+ return undefined;
}
- return action;
-};
-
-const formButtonQuery = 'button[type=\'button\'], button[type=\'submit\'], input[type=\'button\'], button:not([type]), div[role=\'button\']';
+ let usernameExists = false;
+ for (const c of credentials) {
+ if (c.login === usernameValue && c.password === passwordValue) {
+ return false;
+ }
-// Get the form submit button instead if action URL is same as the page itself
-kpxc.getFormSubmitButton = function(form) {
- const action = kpxc.submitUrl || form.action;
- if (action.includes(document.location.origin + document.location.pathname)) {
- for (const i of form.elements) {
- if (i.type === 'submit') {
- return i;
- }
+ if (c.login === usernameValue) {
+ usernameExists = true;
+ break;
}
}
- // Try to find another button. Select the first one.
- const buttons = Array.from(form.querySelectorAll(formButtonQuery));
- if (buttons.length > 0) {
- return buttons[0];
+ const credentialsList = [];
+ for (const c of credentials) {
+ credentialsList.push({
+ login: c.login,
+ name: c.name,
+ uuid: c.uuid
+ });
}
- // Try to find similar buttons outside the form which are added via 'form' property
- for (const e of form.elements) {
- if ((e.nodeName === 'BUTTON' && e.type === 'button')
- || (e.nodeName === 'BUTTON' && e.type === 'submit')
- || (e.nodeName === 'INPUT' && e.type === 'button')
- || (e.nodeName === 'BUTTON' && e.type === '')) {
- return e;
+ const getUrl = function() {
+ let url = kpxc.settings.saveDomainOnlyNewCreds ? document.location.origin : document.location.href;
+ if (url.indexOf('?') > 0) {
+ url = url.substring(0, url.indexOf('?'));
+ if (url.length < document.location.origin.length) {
+ url = document.location.origin;
+ }
}
- }
- return undefined;
-};
+ return url;
+ };
-kpxc.fillInCredentials = async function(combination, onlyPassword, suppressWarnings) {
- const action = kpxc.getFormActionUrl(combination);
- const u = _f(combination.username);
- const p = _f(combination.password);
+ urlValue = urlValue || getUrl();
- if (combination.isNew) {
- // Initialize form-submit for remembering credentials
- const fieldId = combination.password || combination.username;
- const field = _f(fieldId);
- if (field) {
- const form = kpxc.getForm(field);
- if (form && form.length > 0) {
- kpxcForm.init(form, combination);
+ // Set usernameValue to the first one in the list, or the selected entry
+ if (usernameValue === '') {
+ if (credentialsList.length === 1) {
+ usernameValue = credentialsList[0].login;
+ } else if (credentialsList.length > 1) {
+ const index = await sendMessage('page_get_login_id');
+ if (index >= 0) {
+ usernameValue = credentialsList[index].login;
}
}
}
- if (u) {
- kpxc.u = u;
- }
- if (p) {
- kpxc.p = p;
- }
-
- if (kpxc.url === document.location.href && kpxc.submitUrl === action && kpxc.credentials.length > 0) {
- kpxc.fillIn(combination, onlyPassword, suppressWarnings);
- } else {
- kpxc.url = document.location.href;
- kpxc.submitUrl = action;
-
- const credentials = await browser.runtime.sendMessage({
- action: 'retrieve_credentials',
- args: [ kpxc.url, kpxc.submitUrl, true ] // Sets triggerUnlock to true
- });
+ // Show the Credential Banner
+ kpxcBanner.create({
+ username: usernameValue,
+ password: passwordValue,
+ url: urlValue,
+ usernameExists: usernameExists,
+ list: credentialsList
+ });
- await kpxc.retrieveCredentialsCallback(credentials, true);
- kpxc.fillIn(combination, onlyPassword, suppressWarnings);
- }
+ return true;
};
-kpxc.fillInFromActiveElement = async function(suppressWarnings, passOnly = false) {
+// Save credentials triggered fron the context menu
+kpxc.rememberCredentialsFromContextMenu = async function() {
const el = document.activeElement;
- if (el.tagName.toLowerCase() !== 'input') {
- if (kpxcFields.combinations.length > 0) {
- kpxc.fillInCredentials(kpxcFields.combinations[0], passOnly, suppressWarnings);
-
- // Focus to the input field
- const field = _f(passOnly ? kpxcFields.combinations[0].password : kpxcFields.combinations[0].username);
- if (field) {
- field.focus();
- }
- }
+ if (el.nodeName !== 'INPUT') {
return;
}
- kpxcFields.setUniqueId(el);
- const fieldId = kpxcFields.prepareId(el.getAttribute('data-kpxc-id'));
- let combination = null;
- if (el.getAttribute('type') === 'password') {
- combination = await kpxcFields.getCombination('password', fieldId);
- } else {
- combination = await kpxcFields.getCombination('username', fieldId);
+ const type = el.getAttribute('type');
+ const combination = await kpxcFields.getCombination(el, (type === 'password' ? type : 'username'));
+ const usernameValue = combination.username ? combination.username.value : '';
+ const passwordValue = combination.password ? combination.password.value : '';
+
+ const result = await kpxc.rememberCredentials(usernameValue, passwordValue);
+ if (result === undefined) {
+ kpxcUI.createNotification('error', tr('rememberNoPassword'));
+ return;
}
- if (passOnly) {
- if (!_f(combination.password)) {
- kpxcUI.createNotification('warning', tr('fieldsNoPasswordField'));
- return;
- }
+ if (!result) {
+ kpxcUI.createNotification('warning', tr('rememberCredentialsExists'));
}
+};
- delete combination.loginId;
+// The basic function for retrieving credentials from KeePassXC
+kpxc.retrieveCredentials = async function() {
+ kpxc.url = document.location.href;
+ kpxc.submitUrl = kpxc.getFormActionUrl(kpxc.combinations[0]);
- kpxc.fillInCredentials(combination, passOnly, suppressWarnings);
+ if (kpxc.settings.autoRetrieveCredentials && kpxc.url && kpxc.submitUrl) {
+ await kpxc.retrieveCredentialsCallback(await sendMessage('retrieve_credentials', [ kpxc.url, kpxc.submitUrl ]));
+ }
};
-kpxc.fillInFromActiveElementTOTPOnly = async function(target) {
- const el = target || document.activeElement;
- kpxcFields.setUniqueId(el);
- const fieldId = kpxcFields.prepareId(el.getAttribute('data-kpxc-id'));
+// Handles credentials from 'retrieve_credentials' response
+kpxc.retrieveCredentialsCallback = async function(credentials) {
+ _called.retrieveCredentials = true;
+ if (credentials && credentials.length > 0) {
+ kpxc.credentials = credentials;
+ await kpxc.prepareCredentials();
+ }
- const index = await browser.runtime.sendMessage({
- action: 'page_get_login_id'
- });
+ // Retrieve submitted credentials if available
+ const creds = await sendMessage('page_get_submitted');
+ if (creds && creds.submitted) {
+ await sendMessage('page_clear_submitted');
+ kpxc.rememberCredentials(creds.username, creds.password, creds.url, creds.oldCredentials);
+ }
+};
- if (index >= 0 && kpxc.credentials[index]) {
- // Check the value from stringFields (to be removed)
- const currentField = _fs(fieldId);
- if (kpxc.credentials[index].stringFields && kpxc.credentials[index].stringFields.length > 0) {
- const stringFields = kpxc.credentials[index].stringFields;
- for (const s of stringFields) {
- const val = s['KPH: {TOTP}'];
- if (val) {
- kpxc.setValue(currentField, val);
- }
- }
- } else if (kpxc.credentials[index].totp && kpxc.credentials[index].totp.length > 0) {
- kpxc.setValue(currentField, kpxc.credentials[index].totp);
+// If credentials are not received, request them again
+kpxc.receiveCredentialsIfNecessary = async function() {
+ if (kpxc.credentials.length === 0 && !_called.retrieveCredentials) {
+ if (!kpxc.url) {
+ kpxc.url = document.location.href;
+ }
+
+ // Sets triggerUnlock to true
+ const credentials = await sendMessage('retrieve_credentials', [ kpxc.url, kpxc.submitUrl, true ]);
+ if (credentials.length === 0) {
+ return [];
}
+
+ // If the database was locked, this is scope never met. In these cases the response is met at kpxc.detectDatabaseChange
+ await sendMessage('page_set_manual_fill', ManualFill.NONE);
+ await kpxc.retrieveCredentialsCallback(credentials);
+ return credentials;
}
+
+ return kpxc.credentials;
+};
+
+kpxc.setPasswordFilled = async function(state) {
+ await sendMessage('password_set_filled', state);
};
+// Special handling for settings value to select element
kpxc.setValue = function(field, value) {
if (field.matches('select')) {
value = value.toLowerCase().trim();
const options = field.querySelectorAll('option');
+
for (const o of options) {
if (o.textContent.toLowerCase().trim() === value) {
kpxc.setValueWithChange(field, o.value);
return false;
}
}
- } else {
- kpxc.setValueWithChange(field, value);
- }
-};
-
-kpxc.fillInStringFields = function(fields, stringFields, filledInFields) {
- let filledIn = false;
- filledInFields.list = [];
- if (fields && stringFields && fields.length > 0 && stringFields.length > 0) {
- for (let i = 0; i < fields.length; i++) {
- const currentField = _fs(fields[i]);
- const stringFieldValue = Object.values(stringFields[i]);
- if (currentField && stringFieldValue[0]) {
- kpxc.setValue(currentField, stringFieldValue[0]);
- filledInFields.list.push(fields[i]);
- filledIn = true;
- }
- }
+ return;
}
- return filledIn;
+ kpxc.setValueWithChange(field, value);
};
+// Sets a new value to input field and triggers necessary events
kpxc.setValueWithChange = function(field, value) {
- if (kpxc.settings.respectMaxLength === true) {
- const attributeMaxlength = field.getAttribute('maxlength');
- if (attributeMaxlength && !isNaN(attributeMaxlength) && attributeMaxlength > 0) {
- value = value.substr(0, attributeMaxlength);
- }
- }
-
field.value = value;
field.dispatchEvent(new Event('input', { 'bubbles': true }));
field.dispatchEvent(new Event('change', { 'bubbles': true }));
+ field.dispatchEvent(new KeyboardEvent('keydown', { bubbles: true, cancelable: false, key: '', char: '' }));
+ field.dispatchEvent(new KeyboardEvent('keypress', { bubbles: true, cancelable: false, key: '', char: '' }));
+ field.dispatchEvent(new KeyboardEvent('keyup', { bubbles: true, cancelable: false, key: '', char: '' }));
};
-kpxc.fillWithSpecificLogin = async function(id) {
- if (kpxc.credentials[id]) {
- let combination = null;
- if (kpxc.u) {
- kpxc.setValueWithChange(kpxc.u, kpxc.credentials[id].login);
- combination = await kpxcFields.getCombination('username', kpxc.u);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: id
- });
- kpxc.u.focus();
- }
- if (kpxc.p) {
- kpxc.setValueWithChange(kpxc.p, kpxc.credentials[id].password);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: id
- });
- combination = await kpxcFields.getCombination('password', kpxc.p);
+// Returns true if site is ignored
+kpxc.siteIgnored = async function(condition) {
+ kpxc.initSitePreferences();
+
+ if (kpxc.settings.sitePreferences) {
+ let currentLocation;
+ try {
+ currentLocation = window.top.location.href;
+ } catch (err) {
+ // Cross-domain security error inspecting window.top.location.href.
+ // This catches an error when an iframe is being accessed from another (sub)domain -> use the iframe URL instead.
+ currentLocation = window.self.location.href;
}
- const list = [];
- if (kpxc.fillInStringFields(combination.fields, kpxc.credentials[id].stringFields, list)) {
- kpxcForm.destroy(false, { 'password': list.list[0], 'username': list.list[1] });
+ const currentSetting = condition || IGNORE_FULL;
+ for (const site of kpxc.settings.sitePreferences) {
+ if (siteMatch(site.url, currentLocation) || site.url === currentLocation) {
+ if (site.ignore === currentSetting) {
+ return true;
+ }
+
+ kpxc.singleInputEnabledForPage = site.usernameOnly;
+ }
}
+ }
- kpxcAutocomplete.closeList();
+ return false;
+};
+
+// Updates database status and icons when tab is activated again
+kpxc.triggerActivatedTab = async function() {
+ await kpxc.updateDatabaseState();
+ kpxcIcons.switchIcons();
+
+ if (kpxc.databaseState === DatabaseState.UNLOCKED && kpxc.credentials.length === 0) {
+ await kpxc.retrieveCredentials();
+ } else if (kpxc.credentials.length > 0) {
+ kpxc.initLoginPopup();
}
};
-kpxc.fillIn = function(combination, onlyPassword, suppressWarnings) {
- // No credentials available
- if (kpxc.credentials.length === 0 && !suppressWarnings) {
- kpxcUI.createNotification('error', tr('credentialsNoLoginsFound'));
+// Updates the database state to the content script
+kpxc.updateDatabaseState = async function() {
+ const res = await sendMessage('get_status', [ true ]);
+
+ if (!res.keePassXCAvailable) {
+ kpxc.databaseState = DatabaseState.DISCONNECTED;
return;
}
- let skipAutoSubmit = false;
- const uField = _f(combination.username);
- const pField = _f(combination.password);
-
- // Exactly one pair of credentials available
- if (kpxc.credentials.length === 1) {
- let filledIn = false;
+ kpxc.databaseState = res.databaseClosed ? DatabaseState.LOCKED : DatabaseState.UNLOCKED;
+};
- if (kpxc.credentials[0].skipAutoSubmit !== undefined) {
- skipAutoSubmit = kpxc.credentials[0].skipAutoSubmit === 'true';
- }
- if (uField && (!onlyPassword || _singleInputEnabledForPage)) {
- kpxc.setValueWithChange(uField, kpxc.credentials[0].login);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: 0
- });
- filledIn = true;
- }
- if (pField) {
- pField.setAttribute('type', 'password');
- kpxc.setValueWithChange(pField, kpxc.credentials[0].password);
- pField.setAttribute('unchanged', true);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: 0
- });
- filledIn = true;
- kpxc.setPasswordFilled(true);
- }
+/**
+ * @Object kpxcObserverHelper
+ * MutationObserver handler for dynamically added input fields.
+ */
+const kpxcObserverHelper = {};
+kpxcObserverHelper.ignoredNodeNames = [ 'g', 'path', 'svg', 'A', 'HEAD', 'HTML', 'LABEL', 'LINK', 'SCRIPT', 'SPAN', 'VIDEO' ];
+
+kpxcObserverHelper.ignoredNodeTypes = [
+ Node.ATTRIBUTE_NODE,
+ Node.TEXT_NODE,
+ Node.CDATA_SECTION_NODE,
+ Node.PROCESSING_INSTRUCTION_NODE,
+ Node.COMMENT_NODE,
+ Node.DOCUMENT_TYPE_NODE,
+ Node.NOTATION_NODE
+];
- const list = [];
- if (kpxc.fillInStringFields(combination.fields, kpxc.credentials[0].stringFields, list)) {
- kpxcForm.destroy(false, { 'password': list.list[0], 'username': list.list[1] });
- filledIn = true;
- }
+kpxcObserverHelper.inputTypes = [
+ 'text',
+ 'email',
+ 'password',
+ 'tel',
+ 'number',
+ 'username', // Note: Not a standard
+ undefined, // Input field can be without any type. Include this and null to the list.
+ null
+];
- if (!filledIn) {
- if (!suppressWarnings) {
- kpxcUI.createNotification('error', tr('fieldsFill'));
- }
- return;
- }
- } else if (combination.loginId !== undefined && kpxc.credentials[combination.loginId]) {
- // Specific login ID given
- let filledIn = false;
+// Define what element should be observed by the observer
+// and what types of mutations trigger the callback
+kpxcObserverHelper.observerConfig = {
+ subtree: true,
+ attributes: true,
+ childList: true,
+ characterData: true,
+ attributeFilter: [ 'style', 'class' ]
+};
- if (kpxc.credentials[0].skipAutoSubmit !== undefined) {
- skipAutoSubmit = kpxc.credentials[combination.loginId].skipAutoSubmit === 'true';
- }
+// Stores mutation style to an cache array
+kpxcObserverHelper.cacheStyle = function(mut, styleMutations) {
+ if (mut.attributeName !== 'style') {
+ return;
+ }
- if (uField && (!onlyPassword || _singleInputEnabledForPage)) {
- kpxc.setValueWithChange(uField, kpxc.credentials[combination.loginId].login);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: combination.loginId
- });
- filledIn = true;
- }
+ // If the target is inside a form we are monitoring, calculate the CSS style for better compatibility.
+ // getComputedStyle() is very slow, so we cannot do that for every style target.
+ let style = mut.target.style;
+ if (kpxcForm.formIdentified(mut.target.parentNode)) {
+ style = getComputedStyle(mut.target);
+ }
- if (pField) {
- kpxc.setValueWithChange(pField, kpxc.credentials[combination.loginId].password);
- pField.setAttribute('unchanged', true);
- browser.runtime.sendMessage({
- action: 'page_set_login_id', args: combination.loginId
+ if (style.display || style.zIndex) {
+ if (!styleMutations.some(m => m.target === mut.target)) {
+ styleMutations.push({
+ target: mut.target,
+ display: style.display,
+ zIndex: style.zIndex
});
- filledIn = true;
- kpxc.setPasswordFilled(true);
- }
-
- const list = [];
- if (kpxc.fillInStringFields(combination.fields, kpxc.credentials[combination.loginId].stringFields, list)) {
- kpxcForm.destroy(false, { 'password': list.list[0], 'username': list.list[1] });
- filledIn = true;
- }
-
- if (!filledIn) {
- if (!suppressWarnings) {
- kpxcUI.createNotification('error', tr('fieldsFill'));
+ } else {
+ const currentStyle = styleMutations.find(m => m.target === mut.target);
+ if (currentStyle
+ && (currentStyle.display !== style.display
+ || currentStyle.zIndex !== style.zIndex)) {
+ currentStyle.display = style.display;
+ currentStyle.zIndex = style.zIndex;
}
- return;
}
- } else { // Multiple credentials available
- // Check if only one password for given username exists
- let countPasswords = 0;
-
- if (uField) {
- let valPassword = '';
- let valUsername = '';
- let valStringFields = [];
- const valQueryUsername = uField.value.toLowerCase();
-
- // Find passwords to given username (even those with empty username)
- for (const c of kpxc.credentials) {
- if (c.login.toLowerCase() === valQueryUsername) {
- countPasswords += 1;
- valPassword = c.password;
- valUsername = c.login;
- valStringFields = c.stringFields;
-
- if (c.skipAutoSubmit !== undefined) {
- skipAutoSubmit = c.skipAutoSubmit === 'true';
- }
- }
- }
+ }
+};
- // For the correct notification message: 0 = no logins, X > 1 = too many logins
- if (countPasswords === 0) {
- countPasswords = kpxc.credentials.length;
- }
+// Gets input fields from the target
+kpxcObserverHelper.getInputs = function(target, ignoreVisibility = false) {
+ // Ignores target element if it's not an element node
+ if (kpxcObserverHelper.ignoredNode(target)) {
+ return [];
+ }
- // Only one mapping username found
- if (countPasswords === 1) {
- if (!onlyPassword) {
- kpxc.setValueWithChange(uField, valUsername);
- }
+ // Filter out any input fields with type 'hidden' right away
+ const inputFields = [];
+ Array.from(target.getElementsByTagName('input')).forEach(e => {
+ if (e.type !== 'hidden' && !e.disabled && !kpxcObserverHelper.alreadyIdentified(e)) {
+ inputFields.push(e);
+ }
+ });
- if (pField) {
- kpxc.setValueWithChange(pField, valPassword);
- pField.setAttribute('unchanged', true);
- kpxc.setPasswordFilled(true);
- }
+ if (inputFields.length === 0) {
+ return [];
+ }
- const list = [];
- if (kpxc.fillInStringFields(combination.fields, valStringFields, list)) {
- kpxcForm.destroy(false, { 'password': list.list[0], 'username': list.list[1] });
- }
- }
+ // Do not allow more visible inputs than _maximumInputs (default value: 100) -> return the first 100
+ if (inputFields.length > _maximumInputs) {
+ return inputFields.slice(0, _maximumInputs);
+ }
- // User has to select correct credentials by himself
- if (countPasswords > 1) {
- if (!suppressWarnings) {
- const target = onlyPassword ? pField : uField;
- if (!target) {
- return;
- }
-
- if (kpxcAutocomplete.started) {
- kpxcAutocomplete.showList(target);
- } else {
- kpxcAutocomplete.create(target, true, kpxc.settings.autoSubmit);
- }
- target.focus();
- }
- return;
- } else if (countPasswords < 1) {
- if (!suppressWarnings) {
- kpxcUI.createNotification('error', tr('credentialsNoUsernameFound'));
- }
- return;
- }
- } else {
- if (!suppressWarnings) {
- const target = onlyPassword ? pField : uField;
- if (!target) {
- return;
- }
+ // Only include input fields that match with kpxcObserverHelper.inputTypes
+ const inputs = [];
+ for (const field of inputFields) {
+ if (!ignoreVisibility && !kpxcFields.isVisible(field)) {
+ continue;
+ }
- if (kpxcAutocomplete.started) {
- kpxcAutocomplete.showList(target);
- } else {
- kpxcAutocomplete.create(target, true, kpxc.settings.autoSubmit);
- }
- target.focus();
- return;
- }
+ const type = field.getLowerCaseAttribute('type');
+ if (kpxcObserverHelper.inputTypes.includes(type)) {
+ inputs.push(field);
}
}
- // Close autocomplete menu after fill
- kpxcAutocomplete.closeList();
+ return inputs;
+};
- // Auto-submit
- if (kpxc.settings.autoSubmit && !skipAutoSubmit) {
- const form = kpxc.u.form || kpxc.p.form;
- const submitButton = kpxc.getFormSubmitButton(form);
- if (submitButton !== undefined) {
- submitButton.click();
- } else {
- form.submit();
- }
- }
+// Checks if the input field has already identified at page load
+kpxcObserverHelper.alreadyIdentified = function(target) {
+ return kpxc.inputs.some(e => e === target);
};
-kpxc.contextMenuRememberCredentials = async function() {
- const el = document.activeElement;
- if (el.tagName.toLowerCase() !== 'input') {
+// Adds elements to a monitor array. Identifies the input fields.
+kpxcObserverHelper.handleObserverAdd = async function(target) {
+ if (kpxcObserverHelper.ignoredElement(target)) {
return;
}
- kpxcFields.setUniqueId(el);
- const fieldId = kpxcFields.prepareId(el.getAttribute('data-kpxc-id'));
- let combination = null;
- if (el.getAttribute('type') === 'password') {
- combination = await kpxcFields.getCombination('password', fieldId);
- } else {
- combination = await kpxcFields.getCombination('username', fieldId);
+ // Sometimes the settings haven't been loaded before new input fields are detected
+ if (Object.keys(kpxc.settings).length === 0) {
+ kpxc.init();
+ return;
}
- let usernameValue = '';
- let passwordValue = '';
+ const inputs = kpxcObserverHelper.getInputs(target);
+ if (inputs.length === 0) {
+ return;
+ }
- const usernameField = _f(combination.username);
- const passwordField = _f(combination.password);
+ await kpxc.initCombinations(inputs);
+ await kpxcIcons.initIcons(kpxc.combinations);
- if (usernameField) {
- usernameValue = usernameField.value;
+ if (kpxc.databaseState === DatabaseState.UNLOCKED && _called.retrieveCredentials === false) {
+ await kpxc.retrieveCredentials();
}
+};
- if (passwordField) {
- passwordValue = passwordField.value;
+// Removes monitored elements
+kpxcObserverHelper.handleObserverRemove = function(target) {
+ if (kpxcObserverHelper.ignoredElement(target)) {
+ return;
}
- const result = await kpxc.rememberCredentials(usernameValue, passwordValue);
- if (result === undefined) {
- kpxcUI.createNotification('error', tr('rememberNoPassword'));
+ const inputs = kpxcObserverHelper.getInputs(target, true);
+ if (inputs.length === 0) {
return;
}
- if (!result) {
- kpxcUI.createNotification('warning', tr('rememberCredentialsExists'));
- }
+ kpxcIcons.deleteHiddenIcons();
};
-/**
- * Gets the credential list and shows the update banner
- * @param {string} usernameValue Submitted username
- * @param {string} passwordValue Submitted password
- * @param {string} urlValue URL of the page where password change was detected
- * @param {Array} oldCredentials Credentials saved from the password change page, if available
- */
-kpxc.rememberCredentials = async function(usernameValue, passwordValue, urlValue, oldCredentials) {
- const credentials = (oldCredentials !== undefined && oldCredentials.length > 0) ? oldCredentials : kpxc.credentials;
-
- // No password given or field cleaned by a site-running script
- // --> no password to save
- if (passwordValue === '') {
- return undefined;
+// Handles CSS transitionend event
+kpxcObserverHelper.handleTransitionEnd = function(e) {
+ if (!e.isTrusted) {
+ return;
}
- let usernameExists = false;
- let nothingChanged = false;
-
- for (const c of credentials) {
- if (c.login === usernameValue && c.password === passwordValue) {
- nothingChanged = true;
- break;
- }
+ kpxcObserverHelper.handleObserverAdd(e.currentTarget);
+};
- if (c.login === usernameValue) {
- usernameExists = true;
- }
+// Returns true if element should be ignored
+kpxcObserverHelper.ignoredElement = function(target) {
+ if (kpxcObserverHelper.ignoredNode(target)) {
+ return true;
}
- if (!nothingChanged) {
- if (!usernameExists) {
- for (const c of credentials) {
- if (c.login === usernameValue) {
- usernameExists = true;
- break;
- }
- }
- }
- const credentialsList = [];
- for (const c of credentials) {
- credentialsList.push({
- login: c.login,
- name: c.name,
- uuid: c.uuid
- });
- }
-
- let url = this.action;
- if (!url) {
- url = kpxc.settings.saveDomainOnlyNewCreds ? document.location.origin : document.location.href;
- if (url.indexOf('?') > 0) {
- url = url.substring(0, url.indexOf('?'));
- if (url.length < document.location.origin.length) {
- url = document.location.origin;
- }
- }
- }
-
- urlValue = urlValue || url;
-
- // Set usernameValue to the first one in the list, or the selected entry
- if (usernameValue === '') {
- if (credentialsList.length === 1) {
- usernameValue = credentialsList[0].login;
- } else if (credentialsList.length > 1) {
- const index = await browser.runtime.sendMessage({
- action: 'page_get_login_id'
- });
-
- if (index >= 0) {
- usernameValue = credentialsList[index].login;
- }
- }
- }
-
- // Show the banner
- const newCredentials = {
- username: usernameValue,
- password: passwordValue,
- url: urlValue,
- usernameExists: usernameExists,
- list: credentialsList
- };
-
- kpxcBanner.create(newCredentials);
+ // Ignore elements that do not have a className (including SVG)
+ if (typeof target.className !== 'string') {
return true;
}
return false;
};
-kpxc.getSite = function(sites) {
- if (!sites || sites.length === 0) {
- return '';
- }
-
- let site = trimURL(sites[0]);
- kpxc.initializeSitePreferences();
-
- if (slashNeededForUrl(site)) {
- site += '/';
- }
+// Ignores all nodes that doesn't contain elements
+// Also ignore few Youtube-specific custom nodeNames
+kpxcObserverHelper.ignoredNode = function(target) {
+ if (kpxcObserverHelper.ignoredNodeTypes.some(e => e === target.nodeType)
+ || kpxcObserverHelper.ignoredNodeNames.some(e => e === target.nodeName)
+ || target.nodeName.startsWith('YTMUSIC')
+ || target.nodeName.startsWith('YT-')) {
+ return true;
+ }
- return site;
+ return false;
};
-kpxc.ignoreSite = async function(sites) {
- const site = kpxc.getSite(sites);
+// Initializes MutationObserver
+kpxcObserverHelper.initObserver = async function() {
+ kpxc.observer = new MutationObserver(function(mutations, obs) {
+ if (document.visibilityState === 'hidden' || kpxcUI.mouseDown) {
+ return;
+ }
- // Check if the site already exists
- let siteExists = false;
- for (const existingSite of kpxc.settings['sitePreferences']) {
- if (existingSite.url === site) {
- existingSite.ignore = IGNORE_NORMAL;
- siteExists = true;
+ // Limit the maximum number of mutations
+ if (mutations.length > _maximumMutations) {
+ mutations = mutations.slice(0, _maximumMutations);
}
- }
- if (!siteExists) {
- kpxc.settings['sitePreferences'].push({
- url: site,
- ignore: IGNORE_NORMAL,
- usernameOnly: false
- });
- }
+ let styleMutations = [];
+ for (const mut of mutations) {
+ if (kpxcObserverHelper.ignoredNode(mut.target)) {
+ continue;
+ }
- await browser.runtime.sendMessage({
- action: 'save_settings',
- args: [ kpxc.settings ]
- });
-};
+ // Cache style mutations. We only need the last style mutation of the target.
+ kpxcObserverHelper.cacheStyle(mut, styleMutations);
-kpxc.addToSitePreferences = async function(sites) {
- kpxc.initializeSitePreferences();
+ if (mut.type === 'childList') {
+ if (mut.addedNodes.length > 0) {
+ kpxcObserverHelper.handleObserverAdd(mut.addedNodes[0]);
+ } else if (mut.removedNodes.length > 0) {
+ kpxcObserverHelper.handleObserverRemove(mut.removedNodes[0]);
+ }
+ } else if (mut.type === 'attributes' && mut.attributeName === 'class') {
+ // Only accept targets with forms
+ const forms = mut.target.nodeName === 'FORM' ? mut.target : mut.target.getElementsByTagName('form');
+ if (forms.length === 0) {
+ continue;
+ }
- // Returns a predefined URL for certain sites
- const site = kpxcSites.definedURL(trimURL(window.top.location.href));
+ // Listen for possible CSS animations
+ mut.target.removeEventListener('transitionend', kpxcObserverHelper.handleTransitionEnd);
+ mut.target.addEventListener('transitionend', kpxcObserverHelper.handleTransitionEnd);
- // Check if the site already exists -> update the current settings
- let siteExists = false;
- for (const existingSite of kpxc.settings['sitePreferences']) {
- if (existingSite.url === site) {
- existingSite.ignore = IGNORE_NOTHING;
- existingSite.usernameOnly = true;
- siteExists = true;
+ // There's an issue here. We cannot know for sure if the class attribute if added or removed.
+ kpxcObserverHelper.handleObserverAdd(mut.target);
+ }
}
- }
-
- if (!siteExists) {
- kpxc.settings['sitePreferences'].push({
- url: site,
- ignore: IGNORE_NOTHING,
- usernameOnly: true
- });
- }
- await browser.runtime.sendMessage({
- action: 'save_settings',
- args: [ kpxc.settings ]
+ // Handle cached style mutations
+ for (const styleMut of styleMutations) {
+ if (styleMut.display !== 'none' && styleMut.display !== '') {
+ kpxcObserverHelper.handleObserverAdd(styleMut.target);
+ } else {
+ kpxcObserverHelper.handleObserverRemove(styleMut.target);
+ }
+ }
});
- browser.runtime.sendMessage({
- action: 'username_field_detected',
- args: false
- });
+ kpxc.observer.observe(document, kpxcObserverHelper.observerConfig);
};
-// Delete previously created Object if it exists. It will be replaced by an Array
-kpxc.initializeSitePreferences = function() {
- if (kpxc.settings['sitePreferences'] !== undefined && kpxc.settings['sitePreferences'].constructor === Object) {
- delete kpxc.settings['sitePreferences'];
- }
+MutationObserver = window.MutationObserver || window.WebKitMutationObserver;
- if (!kpxc.settings['sitePreferences']) {
- kpxc.settings['sitePreferences'] = [];
- }
-};
+/**
+ * Content script initialization.
+ */
+const initContentScript = async function() {
+ try {
+ const settings = await sendMessage('load_settings');
+ kpxc.settings = settings;
-kpxc.getDocumentLocation = function() {
- return kpxc.settings.saveDomainOnly ? document.location.origin : document.location.href;
-};
+ if (await kpxc.siteIgnored()) {
+ return;
+ }
-// Sets the icons to corresponding database lock status
-kpxc.switchIcons = function() {
- kpxcUsernameIcons.switchIcon(_databaseState);
- kpxcPasswordIcons.switchIcon(_databaseState);
- kpxcTOTPIcons.switchIcon(_databaseState);
-};
+ await kpxc.updateDatabaseState();
+ await kpxc.initCredentialFields();
-kpxc.deleteHiddenIcons = function() {
- kpxcUsernameIcons.deleteHiddenIcons();
- kpxcPasswordIcons.deleteHiddenIcons();
- kpxcTOTPIcons.deleteHiddenIcons();
-};
+ if (kpxc.settings.useObserver) {
+ await kpxcObserverHelper.initObserver();
+ }
-kpxc.setPasswordFilled = function(state) {
- browser.runtime.sendMessage({
- action: 'password_set_filled',
- args: state
- });
-};
+ // Retrieve submitted credentials if available.
+ const [creds, redirectCount] = await Promise.all([
+ await sendMessage('page_get_submitted'),
+ await sendMessage('page_get_redirect_count')
+ ]);
-kpxc.passwordFilled = async function() {
- return await browser.runtime.sendMessage({ action: 'password_get_filled' });
-};
+ if (creds && creds.submitted) {
+ // If username field is not set, wait for credentials in kpxc.retrieveCredentialsCallback.
+ if (!creds.username) {
+ return;
+ }
-kpxc.updateDatabaseState = async function() {
- const res = await browser.runtime.sendMessage({
- action: 'get_status',
- args: [ true ]
- });
+ if (redirectCount >= kpxc.settings.redirectAllowance) {
+ await sendMessage('page_clear_submitted');
+ }
- if (!res.keePassXCAvailable) {
- _databaseState = DatabaseState.DISCONNECTED;
- return;
+ kpxc.rememberCredentials(creds.username, creds.password, creds.url, creds.oldCredentials);
+ }
+ } catch (err) {
+ console.log('initContentScript error: ', err);
}
-
- _databaseState = res.databaseClosed ? DatabaseState.LOCKED : DatabaseState.UNLOCKED;
};
-const kpxcEvents = {};
-
-kpxcEvents.clearCredentials = function() {
- kpxc.credentials = [];
- kpxcAutocomplete.elements = [];
- _called.retrieveCredentials = false;
+if (document.readyState === 'complete' || (document.readyState !== 'loading' && !document.documentElement.doScroll)) {
+ initContentScript();
+} else {
+ document.addEventListener('DOMContentLoaded', initContentScript);
+}
- if (kpxc.settings.autoCompleteUsernames) {
- for (const c of kpxcFields.combinations) {
- const uField = _f(c.username);
- if (uField) {
- if (uField.classList.contains('ui-autocomplete-input')) {
- uField.autocomplete('destroy');
- }
- }
+// These are executed in each frame
+browser.runtime.onMessage.addListener(async function(req, sender) {
+ if ('action' in req) {
+ // Don't allow any actions if the site is ignored
+ if (await kpxc.siteIgnored()) {
+ return;
}
- }
-};
-
-kpxcEvents.triggerActivatedTab = async function() {
- kpxc.init();
- // initCredentialFields calls also "retrieve_credentials", to prevent it
- // check of init() was already called
- if (_called.initCredentialFields && (kpxc.url && kpxc.submitUrl) && kpxc.settings.autoRetrieveCredentials) {
- _called.retrieveCredentials = true;
- kpxc.retrieveCredentialsCallback(await browser.runtime.sendMessage({
- action: 'retrieve_credentials',
- args: [ kpxc.url, kpxc.submitUrl ]
- }));
+ if (req.action === 'activated_tab') {
+ kpxc.triggerActivatedTab();
+ } else if (req.action === 'add_username_only_option') {
+ kpxc.addToSitePreferences();
+ } else if (req.action === 'check_database_hash' && 'hash' in req) {
+ kpxc.detectDatabaseChange(req);
+ } else if (req.action === 'choose_credential_fields') {
+ kpxcDefine.init();
+ } else if (req.action === 'clear_credentials') {
+ kpxc.clearAllFromPage();
+ } else if (req.action === 'fill_user_pass_with_specific_login') {
+ kpxc.fillFromPopup(req.id, req.uuid);
+ } else if (req.action === 'fill_username_password') {
+ sendMessage('page_set_manual_fill', ManualFill.BOTH);
+ await kpxc.receiveCredentialsIfNecessary();
+ kpxc.fillInFromActiveElement();
+ } else if (req.action === 'fill_password') {
+ sendMessage('page_set_manual_fill', ManualFill.PASSWORD);
+ await kpxc.receiveCredentialsIfNecessary();
+ kpxc.fillInFromActiveElement(true); // passOnly to true
+ } else if (req.action === 'fill_totp') {
+ await kpxc.receiveCredentialsIfNecessary();
+ kpxc.fillFromTOTP();
+ } else if (req.action === 'ignore_site') {
+ kpxc.ignoreSite(req.args);
+ } else if (req.action === 'redetect_fields') {
+ const response = await sendMessage('load_settings');
+ kpxc.settings = response;
+ kpxc.inputs = [];
+ kpxc.combinations = [];
+ kpxc.initCredentialFields();
+ } else if (req.action === 'remember_credentials') {
+ kpxc.rememberCredentialsFromContextMenu();
+ } else if (req.action === 'show_password_generator') {
+ kpxcPasswordDialog.trigger();
+ }
}
-};
+});
diff --git a/keepassxc-browser/content/pwgen.js b/keepassxc-browser/content/pwgen.js
index 2fc8267..a9aa134 100644
--- a/keepassxc-browser/content/pwgen.js
+++ b/keepassxc-browser/content/pwgen.js
@@ -3,8 +3,8 @@
const kpxcPasswordIcons = {};
kpxcPasswordIcons.icons = [];
-kpxcPasswordIcons.newIcon = function(useIcons, field, inputs, pos, databaseState = DatabaseState.DISCONNECTED) {
- kpxcPasswordIcons.icons.push(new PasswordIcon(useIcons, field, inputs, pos, databaseState));
+kpxcPasswordIcons.newIcon = function(field, databaseState = DatabaseState.DISCONNECTED) {
+ kpxcPasswordIcons.icons.push(new PasswordIcon(field, databaseState));
};
kpxcPasswordIcons.switchIcon = function(state) {
@@ -15,57 +15,38 @@ kpxcPasswordIcons.deleteHiddenIcons = function() {
kpxcUI.deleteHiddenIcons(kpxcPasswordIcons.icons, 'kpxc-password-field');
};
-
-class PasswordIcon extends Icon {
- constructor(useIcons, field, inputs, pos, databaseState) {
- super();
- this.useIcons = useIcons;
- this.databaseState = databaseState;
-
- if (this.initField(field, inputs, pos)) {
- kpxcUI.monitorIconPosition(this);
- }
- }
-}
-
-PasswordIcon.prototype.initField = function(field, inputs, pos) {
+kpxcPasswordIcons.isValid = function(field) {
if (!field
|| field.readOnly
- || field.offsetWidth < MINIMUM_INPUT_FIELD_WIDTH) {
+ || field.offsetWidth < MINIMUM_INPUT_FIELD_WIDTH
+ || kpxcIcons.hasIcon(field)
+ || !kpxcFields.isVisible(field)) {
return false;
}
- if (field.getAttribute('kpxc-password-field')
- || (field.hasAttribute('kpxc-defined') && field.getAttribute('kpxc-defined') !== 'password')) {
- return false;
- }
+ return true;
+};
- field.setAttribute('kpxc-password-field', true);
- if (this.useIcons) {
- // Observer the visibility
- if (this.observer) {
- this.observer.observe(field);
- }
- this.createIcon(field);
- }
+class PasswordIcon extends Icon {
+ constructor(field, databaseState = DatabaseState.DISCONNECTED) {
+ super();
+ this.databaseState = databaseState;
+ this.nextFieldExists = false;
- this.inputField = field;
+ this.initField(field);
+ kpxcUI.monitorIconPosition(this);
+ }
+}
- let found = false;
- if (inputs) {
- for (let i = pos + 1; i < inputs.length; i++) {
- if (inputs[i] && inputs[i].getLowerCaseAttribute('type') === 'password') {
- field.setAttribute('kpxc-pwgen-next-field-id', inputs[i].getAttribute('data-kpxc-id'));
- field.setAttribute('kpxc-pwgen-next-is-password-field', (i === 0));
- found = true;
- break;
- }
- }
+PasswordIcon.prototype.initField = function(field) {
+ // Observer the visibility
+ if (this.observer) {
+ this.observer.observe(field);
}
- field.setAttribute('kpxc-pwgen-next-field-exists', found);
- return true;
+ this.createIcon(field);
+ this.inputField = field;
};
PasswordIcon.prototype.createIcon = function(field) {
@@ -81,6 +62,7 @@ PasswordIcon.prototype.createIcon = function(field) {
'offset': offset,
'kpxc-pwgen-field-id': field.getAttribute('data-kpxc-id')
});
+
icon.style.zIndex = '10000000';
icon.style.width = Pixels(size);
icon.style.height = Pixels(size);
@@ -116,6 +98,8 @@ PasswordIcon.prototype.createIcon = function(field) {
const kpxcPasswordDialog = {};
kpxcPasswordDialog.created = false;
kpxcPasswordDialog.icon = null;
+kpxcPasswordDialog.input = null;
+kpxcPasswordDialog.nextField = null;
kpxcPasswordDialog.selected = null;
kpxcPasswordDialog.startPosX = 0;
kpxcPasswordDialog.startPosY = 0;
@@ -124,17 +108,6 @@ kpxcPasswordDialog.diffY = 0;
kpxcPasswordDialog.dialog = null;
kpxcPasswordDialog.titleBar = null;
-kpxcPasswordDialog.removeIcon = function(field) {
- if (field.getAttribute('kpxc-password-field')) {
- const pwgenIcons = document.querySelectorAll('.kpxc-pwgen-icon');
- for (const i of pwgenIcons) {
- if (i.getAttribute('kpxc-pwgen-field-id') === field.getAttribute('data-kpxc-id')) {
- document.body.removeChild(i);
- }
- }
- }
-};
-
kpxcPasswordDialog.createDialog = function() {
if (kpxcPasswordDialog.created) {
// If database is open again, generate a new password right away
@@ -245,6 +218,15 @@ kpxcPasswordDialog.showDialog = function(field, icon) {
return;
}
+ kpxcPasswordDialog.input = field;
+
+ // Save next password field if found
+ if (kpxc.inputs.length > 0) {
+ const index = kpxc.inputs.indexOf(field);
+ const nextField = kpxc.inputs[index+1];
+ kpxcPasswordDialog.nextField = (nextField && nextField.getLowerCaseAttribute('type') === 'password') ? nextField : undefined;
+ }
+
kpxcPasswordDialog.createDialog();
initColorTheme(kpxcPasswordDialog.dialog);
kpxcPasswordDialog.openDialog();
@@ -259,10 +241,6 @@ kpxcPasswordDialog.showDialog = function(field, icon) {
kpxcPasswordDialog.dialog.style.top = Pixels(rect.top + rect.height);
kpxcPasswordDialog.dialog.style.left = Pixels(rect.left);
}
-
- kpxcPasswordDialog.dialog.setAttribute('kpxc-pwgen-field-id', field.getAttribute('data-kpxc-id'));
- kpxcPasswordDialog.dialog.setAttribute('kpxc-pwgen-next-field-id', field.getAttribute('kpxc-pwgen-next-field-id'));
- kpxcPasswordDialog.dialog.setAttribute('kpxc-pwgen-next-is-password-field', field.getAttribute('kpxc-pwgen-next-is-password-field'));
}
};
@@ -275,9 +253,7 @@ kpxcPasswordDialog.generate = async function(e) {
e.preventDefault();
}
- callbackGeneratedPassword(await browser.runtime.sendMessage({
- action: 'generate_password'
- }));
+ callbackGeneratedPassword(await sendMessage('generate_password'));
};
kpxcPasswordDialog.copy = function(e) {
@@ -290,35 +266,27 @@ kpxcPasswordDialog.copy = function(e) {
};
kpxcPasswordDialog.fill = function(e) {
- if (!e.isTrusted) {
+ if (!e.isTrusted || !kpxcPasswordDialog.input) {
return;
}
e.preventDefault();
- // Use the active input field
- const field = _f(kpxcPasswordDialog.dialog.getAttribute('kpxc-pwgen-field-id'));
- if (field) {
- const password = kpxcPasswordDialog.shadowSelector('.kpxc-pwgen-input');
- if (field.getAttribute('maxlength')) {
- if (password.value.length > field.getAttribute('maxlength')) {
- const message = tr('passwordGeneratorErrorTooLong') + '\r\n'
- + tr('passwordGeneratorErrorTooLongCut') + '\r\n' + tr('passwordGeneratorErrorTooLongRemember');
- message.style.whiteSpace = 'pre';
- browser.runtime.sendMessage({
- action: 'show_notification',
- args: [ message ]
- });
- return;
- }
+ const password = kpxcPasswordDialog.shadowSelector('.kpxc-pwgen-input');
+ if (kpxcPasswordDialog.input.getAttribute('maxlength')) {
+ if (password.value.length > kpxcPasswordDialog.input.getAttribute('maxlength')) {
+ const message = tr('passwordGeneratorErrorTooLong') + '\r\n'
+ + tr('passwordGeneratorErrorTooLongCut') + '\r\n' + tr('passwordGeneratorErrorTooLongRemember');
+ message.style.whiteSpace = 'pre';
+ sendMessage('show_notification' [ message ]);
+ return;
}
+ }
- field.value = password.value;
- const nextFieldId = field.getAttribute('kpxc-pwgen-next-field-id');
- const nextField = $('input[data-kpxc-id=\'' + nextFieldId + '\']');
- if (nextField) {
- nextField.value = password.value;
- }
+ kpxcPasswordDialog.input.value = password.value;
+
+ if (kpxcPasswordDialog.nextField) {
+ kpxcPasswordDialog.nextField.value = password.value;
}
};
diff --git a/keepassxc-browser/content/totp-field.js b/keepassxc-browser/content/totp-field.js
index e09595c..f811e79 100644
--- a/keepassxc-browser/content/totp-field.js
+++ b/keepassxc-browser/content/totp-field.js
@@ -3,6 +3,17 @@
const ignoreRegex = /(zip|postal).*code/i;
const ignoredTypes = [ 'email', 'password', 'username' ];
+const acceptedOTPFields = [
+ '2fa',
+ 'auth',
+ 'challenge',
+ 'code',
+ 'mfa',
+ 'otp',
+ 'token',
+ 'twofactor'
+];
+
var kpxcTOTPIcons = {};
kpxcTOTPIcons.icons = [];
@@ -18,23 +29,22 @@ kpxcTOTPIcons.deleteHiddenIcons = function() {
kpxcUI.deleteHiddenIcons(kpxcTOTPIcons.icons, 'kpxc-totp-field');
};
+// Quick check for a valid TOTP field
+kpxcTOTPIcons.isAcceptedTOTPField = function(field) {
+ const id = field.getLowerCaseAttribute('id');
+ const name = field.getLowerCaseAttribute('name');
+ const autocomplete = field.getLowerCaseAttribute('autocomplete');
-class TOTPFieldIcon extends Icon {
- constructor(field, databaseState = DatabaseState.DISCONNECTED, forced = false) {
- super();
- this.icon = null;
- this.inputField = null;
- this.databaseState = databaseState;
-
- if (this.initField(field, forced)) {
- kpxcUI.monitorIconPosition(this);
- }
+ if (autocomplete === 'one-time-code' || acceptedOTPFields.some(f => (id && id.includes(f)) || (name && name.includes(f)))) {
+ return true;
}
-}
-TOTPFieldIcon.prototype.initField = function(field, forced) {
- if (!field) {
- return;
+ return false;
+};
+
+kpxcTOTPIcons.isValid = function(field, forced) {
+ if (!field || !kpxcTOTPIcons.isAcceptedTOTPField(field)) {
+ return false;
}
if (!forced) {
@@ -56,8 +66,22 @@ TOTPFieldIcon.prototype.initField = function(field, forced) {
}
}
- field.setAttribute('kpxc-totp-field', 'true');
+ return true;
+};
+class TOTPFieldIcon extends Icon {
+ constructor(field, databaseState = DatabaseState.DISCONNECTED, forced = false) {
+ super();
+ this.icon = null;
+ this.inputField = null;
+ this.databaseState = databaseState;
+
+ this.initField(field, forced);
+ kpxcUI.monitorIconPosition(this);
+ }
+}
+
+TOTPFieldIcon.prototype.initField = function(field, forced) {
// Observer the visibility
if (this.observer) {
this.observer.observe(field);
@@ -65,7 +89,6 @@ TOTPFieldIcon.prototype.initField = function(field, forced) {
this.createIcon(field);
this.inputField = field;
- return false;
};
TOTPFieldIcon.prototype.createIcon = function(field) {
@@ -97,7 +120,7 @@ TOTPFieldIcon.prototype.createIcon = function(field) {
e.preventDefault();
await kpxc.receiveCredentialsIfNecessary();
- kpxc.fillInFromActiveElementTOTPOnly(field);
+ kpxc.fillFromTOTP(field);
});
kpxcUI.setIconPosition(icon, field);
diff --git a/keepassxc-browser/content/ui.js b/keepassxc-browser/content/ui.js
index a7e7c3a..2c70ab8 100644
--- a/keepassxc-browser/content/ui.js
+++ b/keepassxc-browser/content/ui.js
@@ -2,6 +2,12 @@
const MINIMUM_INPUT_FIELD_WIDTH = 60;
+const DatabaseState = {
+ DISCONNECTED: 0,
+ LOCKED: 1,
+ UNLOCKED: 2
+};
+
// jQuery style wrapper for querySelector()
const $ = function(elem) {
return document.querySelector(elem);
@@ -44,6 +50,8 @@ class Icon {
}
const kpxcUI = {};
+kpxcUI.bodyRect = document.body.getBoundingClientRect();
+kpxcUI.bodyStyle = getComputedStyle(document.body);
kpxcUI.mouseDown = false;
// Wrapper for creating elements
@@ -95,14 +103,12 @@ kpxcUI.calculateIconOffset = function(field, size) {
kpxcUI.setIconPosition = function(icon, field) {
const rect = field.getBoundingClientRect();
- const bodyRect = document.body.getBoundingClientRect();
- const bodyStyle = getComputedStyle(document.body);
const size = (document.dir !== 'rtl') ? Number(icon.getAttribute('size')) : 0;
const offset = kpxcUI.calculateIconOffset(field, size);
- if (bodyStyle.position.toLowerCase() === 'relative') {
- icon.style.top = Pixels(rect.top - bodyRect.top + document.scrollingElement.scrollTop + offset + 1);
- icon.style.left = Pixels(rect.left - bodyRect.left + document.scrollingElement.scrollLeft + field.offsetWidth - size - offset);
+ if (kpxcUI.bodyStyle.position.toLowerCase() === 'relative') {
+ icon.style.top = Pixels(rect.top - kpxcUI.bodyRect.top + document.scrollingElement.scrollTop + offset + 1);
+ icon.style.left = Pixels(rect.left - kpxcUI.bodyRect.left + document.scrollingElement.scrollLeft + field.offsetWidth - size - offset);
} else {
icon.style.top = Pixels(rect.top + document.scrollingElement.scrollTop + offset + 1);
icon.style.left = Pixels(rect.left + document.scrollingElement.scrollLeft + field.offsetWidth - size - offset);
@@ -110,11 +116,21 @@ kpxcUI.setIconPosition = function(icon, field) {
};
kpxcUI.deleteHiddenIcons = function(iconList, attr) {
+ const deletedIcons = [];
for (const icon of iconList) {
if (icon.inputField && !kpxcFields.isVisible(icon.inputField)) {
const index = iconList.indexOf(icon);
icon.removeIcon(attr);
iconList.splice(index, 1);
+ deletedIcons.push(icon.inputField);
+ }
+ }
+
+ // Remove the same icons from kpxcIcons.icons array
+ for (const input of deletedIcons) {
+ const index = kpxcIcons.icons.findIndex(e => e.field === input);
+ if (index >= 0) {
+ kpxcIcons.icons.splice(index, 1);
}
}
};
@@ -139,7 +155,7 @@ kpxcUI.updateFromIntersectionObserver = function(iconClass, entries) {
// Wait for possible DOM animations
setTimeout(() => {
kpxcUI.setIconPosition(iconClass.icon, entry.target);
- }, 500);
+ }, 400);
}
}
};
diff --git a/keepassxc-browser/content/username-field.js b/keepassxc-browser/content/username-field.js
index fc5e6b6..6074c1f 100644
--- a/keepassxc-browser/content/username-field.js
+++ b/keepassxc-browser/content/username-field.js
@@ -2,6 +2,7 @@
const kpxcUsernameIcons = {};
kpxcUsernameIcons.icons = [];
+kpxcUsernameIcons.detectedFields = [];
kpxcUsernameIcons.newIcon = function(field, databaseState = DatabaseState.DISCONNECTED) {
kpxcUsernameIcons.icons.push(new UsernameFieldIcon(field, databaseState));
@@ -15,6 +16,18 @@ kpxcUsernameIcons.deleteHiddenIcons = function() {
kpxcUI.deleteHiddenIcons(kpxcUsernameIcons.icons, 'kpxc-username-field');
};
+kpxcUsernameIcons.isValid = function(field) {
+ if (!field
+ || field.offsetWidth < MINIMUM_INPUT_FIELD_WIDTH
+ || field.readOnly
+ || kpxcIcons.hasIcon(field)
+ || !kpxcFields.isVisible(field)) {
+ return false;
+ }
+
+ return true;
+};
+
class UsernameFieldIcon extends Icon {
constructor(field, databaseState = DatabaseState.DISCONNECTED) {
@@ -23,14 +36,15 @@ class UsernameFieldIcon extends Icon {
this.icon = null;
this.inputField = null;
- if (this.initField(field)) {
- kpxcUI.monitorIconPosition(this);
- }
+ this.initField(field);
+ kpxcUI.monitorIconPosition(this);
}
switchIcon(state) {
if (!this.icon) {
return;
+ } else {
+ this.observer.disconnect();
}
this.icon.classList.remove('lock', 'lock-moz', 'unlock', 'unlock-moz', 'disconnected', 'disconnected-moz');
@@ -40,18 +54,6 @@ class UsernameFieldIcon extends Icon {
}
UsernameFieldIcon.prototype.initField = function(field) {
- if (!field
- || field.offsetWidth < MINIMUM_INPUT_FIELD_WIDTH
- || field.readOnly
- || field.getAttribute('kpxc-username-field') === 'true'
- || field.getAttribute('kpxc-totp-field') === 'true'
- || (field.hasAttribute('kpxc-defined') && field.getAttribute('kpxc-defined') !== 'username')
- || !kpxcFields.isVisible(field)) {
- return false;
- }
-
- field.setAttribute('kpxc-username-field', 'true');
-
// Observer the visibility
if (this.observer) {
this.observer.observe(field);
@@ -59,15 +61,9 @@ UsernameFieldIcon.prototype.initField = function(field) {
this.createIcon(field);
this.inputField = field;
- return true;
};
UsernameFieldIcon.prototype.createIcon = function(target) {
- // Remove any existing password generator icons from the input field
- if (target.getAttribute('kpxc-password-field')) {
- kpxcPasswordDialog.removeIcon(target);
- }
-
const field = target;
const className = getIconClassName(this.databaseState);
@@ -124,20 +120,18 @@ const iconClicked = async function(field, icon) {
return;
}
- const connected = await browser.runtime.sendMessage({ action: 'is_connected' });
+ const connected = await sendMessage('is_connected');
if (!connected) {
kpxcUI.createNotification('error', tr('errorNotConnected'));
return;
}
- const databaseHash = await browser.runtime.sendMessage({ action: 'check_database_hash' });
+ const databaseHash = await sendMessage('check_database_hash');
if (databaseHash === '') {
// Triggers database unlock
- _called.manualFillRequested = ManualFill.BOTH;
- await browser.runtime.sendMessage({
- action: 'get_database_hash',
- args: [ false, true ] // Set triggerUnlock to true
- });
+ await sendMessage('page_set_manual_fill', ManualFill.BOTH);
+ await sendMessage('get_database_hash', [ false, true ]); // Set triggerUnlock to true
+ field.focus();
}
if (icon.className.includes('unlock')) {
@@ -165,11 +159,6 @@ const getIconText = function(state) {
};
const fillCredentials = async function(field) {
- const fieldId = field.getAttribute('data-kpxc-id');
- kpxcFields.prepareId(fieldId);
-
- const givenType = field.type === 'password' ? 'password' : 'username';
- const combination = await kpxcFields.getCombination(givenType, fieldId);
-
- kpxc.fillInCredentials(combination, givenType === 'password', false);
+ const combination = await kpxcFields.getCombination(field);
+ kpxc.fillFromUsernameIcon(combination);
};
diff --git a/keepassxc-browser/global.js b/keepassxc-browser/global.js
index f2c392d..1044efd 100755
--- a/keepassxc-browser/global.js
+++ b/keepassxc-browser/global.js
@@ -32,6 +32,12 @@ const AssociatedAction = {
CANCELED: 3
};
+const ManualFill = {
+ NONE: 0,
+ PASSWORD: 1,
+ BOTH: 2
+};
+
/**
* Transforms a valid match pattern into a regular expression
* which matches all URLs included by that pattern.
diff --git a/keepassxc-browser/manifest.json b/keepassxc-browser/manifest.json
index 60e7122..1eacae1 100755
--- a/keepassxc-browser/manifest.json
+++ b/keepassxc-browser/manifest.json
@@ -1,8 +1,8 @@
{
"manifest_version": 2,
"name": "KeePassXC-Browser",
- "version": "1.6.6",
- "version_name": "1.6.6",
+ "version": "1.7.0",
+ "version_name": "1.7.0",
"description": "__MSG_extensionDescription__",
"author": "KeePassXC Team",
"icons": {
diff --git a/keepassxc-browser/options/options.js b/keepassxc-browser/options/options.js
index 0fae4a0..79deb08 100644
--- a/keepassxc-browser/options/options.js
+++ b/keepassxc-browser/options/options.js
@@ -48,8 +48,6 @@ options.saveSetting = async function(name) {
await browser.runtime.sendMessage({
action: 'load_settings'
});
-
- return Promise.resolve();
};
options.saveSettings = async function() {
@@ -66,8 +64,6 @@ options.saveKeyRing = async function() {
await browser.runtime.sendMessage({
action: 'load_keyring'
});
-
- return Promise.resolve();
};
options.initGeneralSettings = function() {
@@ -175,10 +171,8 @@ options.initGeneralSettings = function() {
$('#defaultGroupButton').click(async function() {
const value = $('#defaultGroup').val();
- if (value.length > 0) {
- options.settings['defaultGroup'] = value;
- await options.saveSettings();
- }
+ options.settings['defaultGroup'] = (value.length > 0 ? value : '');
+ await options.saveSettings();
});
$('#defaultGroupButtonReset').click(async function() {
diff --git a/keepassxc-browser/popups/popup.js b/keepassxc-browser/popups/popup.js
index 9846708..bb626f8 100644
--- a/keepassxc-browser/popups/popup.js
+++ b/keepassxc-browser/popups/popup.js
@@ -93,7 +93,7 @@ $(async () => {
$('#lock-database-button').click(async () => {
statusResponse(await browser.runtime.sendMessage({
- action: 'lock-database'
+ action: 'lock_database'
}));
});
diff --git a/keepassxc-browser/popups/popup_httpauth.js b/keepassxc-browser/popups/popup_httpauth.js
index 6d87397..b0383c6 100644
--- a/keepassxc-browser/popups/popup_httpauth.js
+++ b/keepassxc-browser/popups/popup_httpauth.js
@@ -33,7 +33,7 @@ $(async () => {
$('#lock-database-button').click(function() {
browser.runtime.sendMessage({
- action: 'lock-database'
+ action: 'lock_database'
}).then(statusResponse);
});
diff --git a/keepassxc-browser/popups/popup_login.js b/keepassxc-browser/popups/popup_login.js
index 884471b..b79e8bf 100644
--- a/keepassxc-browser/popups/popup_login.js
+++ b/keepassxc-browser/popups/popup_login.js
@@ -8,15 +8,18 @@ $(async () => {
if (tabs.length === 0) {
return; // For example: only the background devtools or a popup are opened
}
- const tab = tabs[0];
+ const tab = tabs[0];
const logins = global.page.tabs[tab.id].loginList;
const ll = document.getElementById('login-list');
+
for (let i = 0; i < logins.length; i++) {
+ const uuid = logins[i].uuid;
const a = document.createElement('a');
- a.textContent = logins[i];
+ a.textContent = logins[i].text;
a.setAttribute('class', 'list-group-item');
a.setAttribute('id', '' + i);
+
a.addEventListener('click', (e) => {
if (!e.isTrusted) {
return;
@@ -25,10 +28,13 @@ $(async () => {
const id = e.target.id;
browser.tabs.sendMessage(tab.id, {
action: 'fill_user_pass_with_specific_login',
- id: Number(id)
+ id: Number(id),
+ uuid: uuid
});
+
close();
});
+
ll.appendChild(a);
}
@@ -50,12 +56,13 @@ $(async () => {
}
}
});
+
filter.focus();
}
$('#lock-database-button').click((e) => {
browser.runtime.sendMessage({
- action: 'lock-database'
+ action: 'lock_database'
});
$('#credentialsList').hide();
$('#database-not-opened').show();
diff --git a/keepassxc-browser/popups/popup_multiple-fields.html b/keepassxc-browser/popups/popup_multiple-fields.html
deleted file mode 100644
index 24dc694..0000000
--- a/keepassxc-browser/popups/popup_multiple-fields.html
+++ /dev/null
@@ -1,39 +0,0 @@
-<!DOCTYPE html>
-<html>
- <head>
- <title data-i18n="popupTitle"></title>
- <meta charset="UTF-8">
- <link rel="stylesheet" href="../css/colors.css" />
- <link rel="stylesheet" href="popup.css" />
- <link rel="stylesheet" href="../bootstrap/bootstrap.min.css" />
- <link rel="stylesheet" href="../fonts/fork-awesome.min.css" />
- <script src="../browser-polyfill.min.js"></script>
- <script src="../global.js"></script>
- <script src="../bootstrap/jquery-3.4.1.min.js"></script>
- <script src="../bootstrap/bootstrap.min.js"></script>
- <script src="popup_functions.js"></script>
- <script defer src="../translate.js"></script>
- </head>
- <body>
- <div class="container">
- <div id="settings" class="settings">
- <button id="btn-options" class="btn btn-sm btn-success"><i class="fa fa-cog" aria-hidden="true"></i> <span data-i18n="popupSettingsText"></span></button>
- <button id="btn-choose-credential-fields" class="btn btn-sm btn-warning"><i class="fa fa-list-alt" aria-hidden="true"></i> <span data-i18n="popupChooseCredentialsText"></span></button>
- <button id="lock-database-button" class="btn btn-sm btn-danger" data-i18n="[title]lockDatabase"><i class="fa fa-lock" aria-hidden="true"></i></button>
-
- <div id="update-available" class="alert alert-danger">
- <span data-i18n="popupUpdateAvailable"></span>
- <br />
- <a target="_blank" class="alert-link" href="https://keepassxc.org/download"><span data-i18n="popupDownloadNewVersion"></span></a>.
- </div>
- </div>
-
- <div>
- <p>
- <span data-i18n="popupMultiplePasswordFields"></span>
- "<code><span data-i18n="contextMenuFillUsernameAndPassword"></span></code>", "<code><span data-i18n="contextMenuFillPassword"></span></code>"
- </p>
- </div>
- </div>
- </body>
-</html>