mirror of
https://github.com/novnc/noVNC.git
synced 2026-06-02 18:39:41 +00:00
This only reads from localstorage in order to initialize the settings map. After initializaton, reads will return the value from the map. When writing a value, the settings map and the local storage are updated, unless the setting is a default value or derived from the query string. This has a few advantages: 1. Saved settings will not be overridden by settings specified in the query string. This means a setting could be temporarily changed using the query string, but once removed from the query string, the setting would return back to what the user selected. 2. Default values will not be saved. If a user has always used the default value for a setting, then they can move to a new version with different defaults without clearing localstorage. 3. Changes made to localstorage in a session running in a different window will not affect the settings in the current window (until the page is refreshed). Regarding eraseSetting: It is possible that another tab could change the value, leading to an unexpected value change in the tab that deletes. However, this function is currently unused, so this will be evaluted if and when it used.
241 lines
6.8 KiB
JavaScript
241 lines
6.8 KiB
JavaScript
/*
|
|
* noVNC: HTML5 VNC client
|
|
* Copyright (C) 2012 Joel Martin
|
|
* Copyright (C) 2013 NTT corp.
|
|
* Licensed under MPL 2.0 (see LICENSE.txt)
|
|
*
|
|
* See README.md for usage and integration instructions.
|
|
*/
|
|
|
|
import { init_logging as main_init_logging } from '../core/util/logging.js';
|
|
|
|
// init log level reading the logging HTTP param
|
|
export function init_logging (level) {
|
|
"use strict";
|
|
if (typeof level !== "undefined") {
|
|
main_init_logging(level);
|
|
} else {
|
|
var param = document.location.href.match(/logging=([A-Za-z0-9\._\-]*)/);
|
|
main_init_logging(param || undefined);
|
|
}
|
|
};
|
|
|
|
// Read a query string variable
|
|
export function getQueryVar (name, defVal) {
|
|
"use strict";
|
|
var re = new RegExp('.*[?&]' + name + '=([^&#]*)'),
|
|
match = document.location.href.match(re);
|
|
if (typeof defVal === 'undefined') { defVal = null; }
|
|
if (match) {
|
|
return decodeURIComponent(match[1]);
|
|
} else {
|
|
return defVal;
|
|
}
|
|
};
|
|
|
|
// Read a hash fragment variable
|
|
export function getHashVar (name, defVal) {
|
|
"use strict";
|
|
var re = new RegExp('.*[&#]' + name + '=([^&]*)'),
|
|
match = document.location.hash.match(re);
|
|
if (typeof defVal === 'undefined') { defVal = null; }
|
|
if (match) {
|
|
return decodeURIComponent(match[1]);
|
|
} else {
|
|
return defVal;
|
|
}
|
|
};
|
|
|
|
// Read a variable from the fragment or the query string
|
|
// Fragment takes precedence
|
|
export function getConfigVar (name, defVal) {
|
|
"use strict";
|
|
var val = getHashVar(name);
|
|
if (val === null) {
|
|
val = getQueryVar(name, defVal);
|
|
}
|
|
return val;
|
|
};
|
|
|
|
/*
|
|
* Cookie handling. Dervied from: http://www.quirksmode.org/js/cookies.html
|
|
*/
|
|
|
|
// No days means only for this browser session
|
|
export function createCookie (name, value, days) {
|
|
"use strict";
|
|
var date, expires;
|
|
if (days) {
|
|
date = new Date();
|
|
date.setTime(date.getTime() + (days * 24 * 60 * 60 * 1000));
|
|
expires = "; expires=" + date.toGMTString();
|
|
} else {
|
|
expires = "";
|
|
}
|
|
|
|
var secure;
|
|
if (document.location.protocol === "https:") {
|
|
secure = "; secure";
|
|
} else {
|
|
secure = "";
|
|
}
|
|
document.cookie = name + "=" + value + expires + "; path=/" + secure;
|
|
};
|
|
|
|
export function readCookie (name, defaultValue) {
|
|
"use strict";
|
|
var nameEQ = name + "=",
|
|
ca = document.cookie.split(';');
|
|
|
|
for (var i = 0; i < ca.length; i += 1) {
|
|
var c = ca[i];
|
|
while (c.charAt(0) === ' ') { c = c.substring(1, c.length); }
|
|
if (c.indexOf(nameEQ) === 0) { return c.substring(nameEQ.length, c.length); }
|
|
}
|
|
return (typeof defaultValue !== 'undefined') ? defaultValue : null;
|
|
};
|
|
|
|
export function eraseCookie (name) {
|
|
"use strict";
|
|
createCookie(name, "", -1);
|
|
};
|
|
|
|
/*
|
|
* Setting handling.
|
|
*/
|
|
|
|
var settings = {};
|
|
|
|
export function initSettings (callback /*, ...callbackArgs */) {
|
|
"use strict";
|
|
var callbackArgs = Array.prototype.slice.call(arguments, 1);
|
|
if (window.chrome && window.chrome.storage) {
|
|
window.chrome.storage.sync.get(function (cfg) {
|
|
settings = cfg;
|
|
if (callback) {
|
|
callback.apply(this, callbackArgs);
|
|
}
|
|
});
|
|
} else {
|
|
settings = {};
|
|
if (callback) {
|
|
callback.apply(this, callbackArgs);
|
|
}
|
|
}
|
|
};
|
|
|
|
// Update the settings cache, but do not write to permanent storage
|
|
export function setSetting (name, value) {
|
|
settings[name] = value;
|
|
};
|
|
|
|
// No days means only for this browser session
|
|
export function writeSetting (name, value) {
|
|
"use strict";
|
|
if (settings[name] === value) return;
|
|
settings[name] = value;
|
|
if (window.chrome && window.chrome.storage) {
|
|
window.chrome.storage.sync.set(settings);
|
|
} else {
|
|
localStorage.setItem(name, value);
|
|
}
|
|
};
|
|
|
|
export function readSetting (name, defaultValue) {
|
|
"use strict";
|
|
var value;
|
|
if ((name in settings) || (window.chrome && window.chrome.storage)) {
|
|
value = settings[name];
|
|
} else {
|
|
value = localStorage.getItem(name);
|
|
settings[name] = value;
|
|
}
|
|
if (typeof value === "undefined") {
|
|
value = null;
|
|
}
|
|
if (value === null && typeof defaultValue !== "undefined") {
|
|
return defaultValue;
|
|
} else {
|
|
return value;
|
|
}
|
|
};
|
|
|
|
export function eraseSetting (name) {
|
|
"use strict";
|
|
// Deleting here means that next time the setting is read when using local
|
|
// storage, it will be pulled from local storage again.
|
|
// If the setting in local storage is changed (e.g. in another tab)
|
|
// between this delete and the next read, it could lead to an unexpected
|
|
// value change.
|
|
delete settings[name];
|
|
if (window.chrome && window.chrome.storage) {
|
|
window.chrome.storage.sync.remove(name);
|
|
} else {
|
|
localStorage.removeItem(name);
|
|
}
|
|
};
|
|
|
|
export function injectParamIfMissing (path, param, value) {
|
|
// force pretend that we're dealing with a relative path
|
|
// (assume that we wanted an extra if we pass one in)
|
|
path = "/" + path;
|
|
|
|
var elem = document.createElement('a');
|
|
elem.href = path;
|
|
|
|
var param_eq = encodeURIComponent(param) + "=";
|
|
var query;
|
|
if (elem.search) {
|
|
query = elem.search.slice(1).split('&');
|
|
} else {
|
|
query = [];
|
|
}
|
|
|
|
if (!query.some(function (v) { return v.startsWith(param_eq); })) {
|
|
query.push(param_eq + encodeURIComponent(value));
|
|
elem.search = "?" + query.join("&");
|
|
}
|
|
|
|
// some browsers (e.g. IE11) may occasionally omit the leading slash
|
|
// in the elem.pathname string. Handle that case gracefully.
|
|
if (elem.pathname.charAt(0) == "/") {
|
|
return elem.pathname.slice(1) + elem.search + elem.hash;
|
|
} else {
|
|
return elem.pathname + elem.search + elem.hash;
|
|
}
|
|
};
|
|
|
|
// sadly, we can't use the Fetch API until we decide to drop
|
|
// IE11 support or polyfill promises and fetch in IE11.
|
|
// resolve will receive an object on success, while reject
|
|
// will receive either an event or an error on failure.
|
|
export function fetchJSON(path, resolve, reject) {
|
|
// NB: IE11 doesn't support JSON as a responseType
|
|
var req = new XMLHttpRequest();
|
|
req.open('GET', path);
|
|
|
|
req.onload = function () {
|
|
if (req.status === 200) {
|
|
try {
|
|
var resObj = JSON.parse(req.responseText);
|
|
} catch (err) {
|
|
reject(err);
|
|
return;
|
|
}
|
|
resolve(resObj);
|
|
} else {
|
|
reject(new Error("XHR got non-200 status while trying to load '" + path + "': " + req.status));
|
|
}
|
|
};
|
|
|
|
req.onerror = function (evt) {
|
|
reject(new Error("XHR encountered an error while trying to load '" + path + "': " + evt.message));
|
|
};
|
|
|
|
req.ontimeout = function (evt) {
|
|
reject(new Error("XHR timed out while trying to load '" + path + "'"));
|
|
};
|
|
|
|
req.send();
|
|
}
|