Source code

Revision control

Copy as Markdown

Other Tools

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
const { LocalizationHelper } = require("resource://devtools/shared/l10n.js");
const L10N = new LocalizationHelper(
"devtools/client/locales/device.properties"
);
const { RemoteSettings } = ChromeUtils.importESModule(
);
loader.lazyRequireGetter(
this,
"asyncStorage",
"resource://devtools/shared/async-storage.js"
);
const LOCAL_DEVICES = "devtools.devices.local";
/* This is a catalog of common web-enabled devices and their properties,
* intended for (mobile) device emulation.
*
* The properties of a device are:
* - name: brand and model(s).
* - width: viewport width.
* - height: viewport height.
* - pixelRatio: ratio from viewport to physical screen pixels.
* - userAgent: UA string of the device's browser.
* - touch: whether it has a touch screen.
* - os: default OS, such as "ios", "fxos", "android".
*
* The device types are:
* ["phones", "tablets", "laptops", "televisions", "consoles", "watches"].
*
* To propose new devices for the shared catalog, see
*
* You can easily add more devices to this catalog from your own code (e.g. an
* addon) like so:
*
* var myPhone = { name: "My Phone", ... };
* require("devtools/client/shared/devices").addDevice(myPhone, "phones");
*/
// Local devices catalog that addons can add to.
let localDevices;
let localDevicesLoaded = false;
/**
* Load local devices from storage.
*/
async function loadLocalDevices() {
if (localDevicesLoaded) {
return;
}
let devicesJSON = await asyncStorage.getItem(LOCAL_DEVICES);
if (!devicesJSON) {
devicesJSON = "{}";
}
localDevices = JSON.parse(devicesJSON);
localDevicesLoaded = true;
}
/**
* Add a device to the local catalog.
* Returns `true` if the device is added, `false` otherwise.
*/
async function addDevice(device, type = "phones") {
await loadLocalDevices();
let list = localDevices[type];
if (!list) {
list = localDevices[type] = [];
}
// Ensure the new device is has a unique name
const exists = list.some(entry => entry.name == device.name);
if (exists) {
return false;
}
list.push(Object.assign({}, device));
await asyncStorage.setItem(LOCAL_DEVICES, JSON.stringify(localDevices));
return true;
}
/**
* Edit a device from the local catalog.
* Returns `true` if the device is edited, `false` otherwise.
*/
async function editDevice(oldDevice, newDevice, type = "phones") {
await loadLocalDevices();
const list = localDevices[type];
if (!list) {
return false;
}
const index = list.findIndex(entry => entry.name == oldDevice.name);
if (index == -1) {
return false;
}
// Replace old device info with new one
list.splice(index, 1, newDevice);
await asyncStorage.setItem(LOCAL_DEVICES, JSON.stringify(localDevices));
return true;
}
/**
* Remove a device from the local catalog.
* Returns `true` if the device is removed, `false` otherwise.
*/
async function removeDevice(device, type = "phones") {
await loadLocalDevices();
const list = localDevices[type];
if (!list) {
return false;
}
const index = list.findIndex(entry => entry.name == device.name);
if (index == -1) {
return false;
}
list.splice(index, 1);
await asyncStorage.setItem(LOCAL_DEVICES, JSON.stringify(localDevices));
return true;
}
/**
* Remove all local devices. Useful to clear everything when testing.
*/
async function removeLocalDevices() {
await asyncStorage.removeItem(LOCAL_DEVICES);
localDevices = {};
}
/**
* Get the complete devices catalog.
*/
async function getDevices() {
const records = await RemoteSettings("devtools-devices").get();
const devicesByType = new Map();
for (const record of records) {
const { type } = record;
if (!devicesByType.has(type)) {
devicesByType.set(type, []);
}
devicesByType.get(type).push(record);
}
await loadLocalDevices();
for (const type in localDevices) {
if (!devicesByType.has(type)) {
devicesByType.set(type, []);
}
devicesByType.get(type).push(...localDevices[type]);
}
return devicesByType;
}
/**
* Get the localized string for a device type.
*/
function getDeviceString(deviceType) {
return L10N.getStr("device." + deviceType);
}
module.exports = {
addDevice,
editDevice,
removeDevice,
removeLocalDevices,
getDevices,
getDeviceString,
};