Files
palemoon27/devtools/shared/css-angle.js
roytam1 ff0033e97d import changes from `dev' branch of rmottola/Arctic-Fox:
- Bug 1231975 - Part 3: Break a reference cycle between PendingResolution and DNSRequestChild. r=drno (ecf45de535)
- Bug 1231975 - Part 4: Add some logging and simplification in TestNrSocket. r=drno (fa811f7743)
- Bug 1231975 - Part 5: Fix an intermittent failure caused by the NAT simulator erroneously canceling NR_ASYNC_WAIT_READ. r=drno (e436a8cc75)
- var-let (d228288673)
- Bug 1256022 - dom/network slow GC on mochitest fix r=dragana (e0dffd5033)
- Bug 1231130 - added mHadLocalInstance to constructor.r=jaas (c4f6d0c530)
- Bug 1121290: Use "%ls" instead of "%s" in _snwprintf_s format string# r=bsmedberg (e0434aca5a)
- Bug 1206952 - Rename MagicRequest to ByteRangeRequest. r=sicking (6780309aa2)
- Bug 1206952 - Convert PluginStreamListener to use channel->AsyncOpen2(). r=sicking (8f41f3e148)
- Bug 1262335 - Part 2. Remove Android GB/HC defines from OMX. r=snorp (e5b7435d92)
- Bug 956899 - Add a std::condition_variable work-alike; r=froydnj (98076c707e)
- Bug 1254123 - Handle OOM more gracefully in js::ErrorToException. (r=Waldo) (f9a2ef18d1)
- Bug 1255128 - Standard argument coercion in new ArrayBuffer(length). r=nbp. Thanks to snowmantw for tests. (06e5cedd80)
- Bug 1251919 - Nuke Debugger wrappers on failure. (r=shu) (64bc41b1f1)
- Bug 1254190 - Propagate failure of matchAllDebuggeeGlobals() in Debugger. (r=shu) (927bf01ec5)
- Bug 1254172 - Make UnboxedLayout::makeNativeGroup robust to unknownProperties on unboxed type. (r=jandem) (398a6b3aa7)
- Bug 1268213 - BlobImplFile::GetTypeRunnable can be a WorkerMainThreadRunnable, r=khuey (30e4ff4b75)
- Bug 1250523 - Wait for the markup-view to be loaded in browser_markup_load_01.js; r=gl (acf9e9d88b)
- Bug 1230325 - markup-view: skip keyboard shortcuts if any modifier;r=pbrosset (ab974dfac7)
- Bug 1155465 - part1: layout-view: use content-type CSS_VALUE for editors;r=miker (d21e70b811)
- Bug 1155465 - part2: inplace-editor: disable increment on up/down for PLAIN_TEXT;r=miker (5d72e05f63)
- Bug 1241126 - ruleview property: open editor for prop. name on click on ":";r=gl (4889010c34)
- Bug 1128340 - Allow renaming with non-ASCII characters in WebIDE. r=jryans (f9844e3afb)
- Bug 1039482 - Properly position and style the file name edit field in the projecteditor. r=bgrins (4c29b80958)
- Bug 1268231 - Get rid of StopSyncLoopRunnable, r=khuey (29b0a0ed4f)
- Bug 1267904 - Add telemetry for WorkerMainThreadRunnable, r=khuey (970d39bcce)
- Bug 1261317 - part1: inplace-editor: small refactor of keypress event handler;r=pbro (5261bca85a)
- Bug 1261317 - part2: prevent autocomplete on arrow keys in multiline editor;r=pbro (859738623a)
- Bug 1246677 - 1 - Make waitForSuccess work with async functions; r=miker (3927347635)
- Bug 1227810 - split browser_ruleview_authored.js into three tests; r=pbrosset (f6e4760908)
- Bug 1246677 - 2 - Stop using CPOWs in simulateColorPickerChange; r=miker (69a31024cf)
- Bug 1246677 - 3 - Remove all usages of getNode in ruleview tests; r=tromey (2f25bec099)
- Bug 1240813 - Fixed the unhandled promise rejections in browser_rules_colorpicker-* tests; r=ochameau (bd534ec71b)
- Bug 1217328 - let filter editor work on invalid values. r=pbrosset (91602cac94)
- Bug 1225236 - Removed the 360 value limit for the hue-rotate field in the CSS filter popup. r=pbro (9bceb9947d)
- Bug 1223076 - make FilterWidget handle "unset", "initial", and "inherit". r=pbrosset (6542d5cc99)
- Bug 1221156 - make FilterWidget try to preserve URL quoting; r=pbrosset (3e1105c3d5)
- Bug 1226543 - fix URL quoting in CSSFilterEditorWidget.getValueAt. r=pbrosset (1770cc279a)
- Bug 1241527 - 1 - Fix some unhandled rejected promises in colorpicker, cubicbezier and cssfilter ruleview tests; r=gl (059df355be)
- Bug 1241527 - 2 - Use ruleview-changed event to avoid pending requests when browser_rules_search-filter* tests end; r=gl (407d5866b0)
- Bug 1241527 - 3 - Fix typo in hideTooltipAndWaitForRuleviewChanged; r=gl (964270c8c3)
- Bug 1241527 - 4 - Use ruleview-changed event to avoid pending requests in browser_rules_multiple* tests; r=gl (2947abf296)
- Bug 1241527 - 5 - Use ruleview-changed event to fix remaining pending requests in tests; r=gl (5391857e12)
- Bug 1246677 - 4 - Stop using content.getComputedStyle in ruleview tests; r=miker (f61cd8d5aa)
- Bug 1237885 - fix add-rules_01 intermittent by splitting in two tests;r=gl (45e4de6012)
- Bug 1166956 - add valid unit when incrementing CSS value "0";r=tromey (40ca31b909)
- Bug 1241155 - correctly use indexOf in browser_rules_user-agent-styles.js; r=bgrins (0966d23f7e)
- Bug 1229911 - recognize DevToolsUtils.defineLazyGetter and defineLazyModuleGetter. r=miker (876e4b9f3b)
- Bug 1230093 - Make the import-headjs-globals rule store variables correctly; r=Mossop (be3ddcb7fb)
- Bug 1229224: Support more forms of defining globals and make anywhere we import scripts use them too. r=miker (c9a243fcae)
- Bug 1231963 - handle top-level "this.mumble" assignments in eslint; r=mikeratcliffe (be07835449)
- Bug 1224289 - add eslint rule to reject Cu.importGlobalProperties; r=mikeratcliffe (eac54ddcce)
- Bug 1241544 - add documentation for this-top-level-scope eslint rule; r=mikeratcliffe (9d832af29d)
- Bug 1239426 - handle arrow functions in getASTSource; r=mikeratcliffe (889b5ff89c)
- Bug 1242584 - Remove dead code in import-headjs-globals. r=tromey (e7b07c59b3)
- Bug 1229224: Support more forms of defining globals and make anywhere we import scripts use them too. r=miker (cf4a3d4d48)
- Bug 1242584 - import-globals-from should carry over to tests. r=tromey (f1f7d7269a)
- Bug 1224735 - only emit one error per possible CPOW use; r=miker,Ms2ger (621dba76e1)
- Bug 1245916: Unify eslint global discovery rules. r=pbrosset (e80d38e097)
- Bug 1246677 - 7 - Clean remaining ruleview and tests eslint warnings; r=jdescottes (9415147b5b)
- Bug 1246677 - 5 - Get rid of 'content' in ruleview test files; r=jdescottes (d868fde632)
- Bug 1209295 - Ensure browser_rules_add-property-cancel_02.js waits for the correct change notification. r=pbrosset (4099437554)
- Bug 1240778 - Fixed the unhandled promise rejection in browser_rules_add-property_01.js; r=ochameau (838bd1f99c)
- Bug 1246677 - 8 - Use addProperty and remove code duplication; r=gl (2c33059028)
- Bug 1246677 - 9 - Get rid of all remaining _applyingModifications usage in tests; r=ochameau (a9bc3b2495)
- Bug 1143742 - part1: multiline inplace editor: cleanup existing tests;r=gl (2bc96e58d3)
- Bug 1243695 - ensure caret is visible in ruleview prop editor;r=miker (3353a5a77c)
- Bug 1143742 - part2: multiline inplace-editor should support a maxWidth option;r=gl (98809c04cb)
- Bug 1178462 - Cancel inplace editor autocomplete on window blur;r=gl (c4af4c03d2)
- Bug 1261827 - inplace-editor: copyTextStyles should not copy line-height property;r=pbro (1fc8d7d432)
- Bug 1069829 - 1 - Remove a usage of domUtils.cssPropertyIsValid in inplace-editor; r=tromey (7898eed84b)
- Bug 1143742 - part3: multiline inplace-editor autocomplete behavior;r=gl (53b369dccb)
- Bug 1143742 - part4: add textarea to valid targets when copying;r=gl (84af495716)
- Bug 1143742 - part5: fix eslint error in inspector ruleview test;r=bustage (d7a53a6217)
- Bug 1249888 - try/catch SourceMapConsumer to avoid empty rule-view when source map is invalid; r=gl (c78a7a6ac7)
- Bug 1255787 - Do not assume sourceMap appears only in external stylesheets; r=gl (7c69e1e559)
- Bug 1029459 - remove output-parser iteration limit. r=pbrosset (f2438a8642)
- Bug 1250835 - Display swatch for angles in the rules panel. r=miker (87721e80f2)
- Bug 1259777 - Remove unnecessary DOMUtils lazy load in css-angle.js . r=pbro (e199a014d5)
- Bug 1259559 - Units cycling with shift+click persists value in Style editor. r=miker (b7074813e9)
- Bug 1245996 - inspector: fix xul scrollbars stealing focus;r=pbro (bcb19ffa83)
- Bug 1180349 - Increase the timeout for browser_markupview_links_01.js (b67cd0d2be)
- Bug 1253935 - Remove all CPOW usages in styleeditor tests and use ContentTask instead of custom frame-script; r=ochameau (485308e7b1)
- Bug 1257246: Update devtools for eslint 2. r=pbro (c04f7f3046)
- Bug 1264968 part 2 - Allow persisting attributes of xul:window if its owner document is not root. r=enndeakin (ca8182b534)
- Bug 1244948 - silence the 'loaded script twice' warning. r=bz (fa571b837c)
2024-08-30 20:58:51 +08:00

345 lines
9.0 KiB
JavaScript

/* 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 {Cc, Ci} = require("chrome");
const SPECIALVALUES = new Set([
"initial",
"inherit",
"unset"
]);
/**
* This module is used to convert between various angle units.
*
* Usage:
* let {require} = Cu.import("resource://devtools/shared/Loader.jsm", {});
* let {angleUtils} = require("devtools/shared/css-angle");
* let angle = new angleUtils.CssAngle("180deg");
*
* angle.authored === "180deg"
* angle.valid === true
* angle.rad === "3,14rad"
* angle.grad === "200grad"
* angle.turn === "0.5turn"
*
* angle.toString() === "180deg"; // Outputs the angle value and its unit
* // Angle objects can be reused
* angle.newAngle("-1TURN") === "-1TURN"; // true
*/
function CssAngle(angleValue) {
this.newAngle(angleValue);
}
module.exports.angleUtils = {
CssAngle: CssAngle,
classifyAngle: classifyAngle
};
CssAngle.ANGLEUNIT = {
"deg": "deg",
"rad": "rad",
"grad": "grad",
"turn": "turn"
};
CssAngle.prototype = {
_angleUnit: null,
_angleUnitUppercase: false,
// The value as-authored.
authored: null,
// A lower-cased copy of |authored|.
lowerCased: null,
get angleUnit() {
if (this._angleUnit === null) {
this._angleUnit = classifyAngle(this.authored);
}
return this._angleUnit;
},
set angleUnit(unit) {
this._angleUnit = unit;
},
get valid() {
return /^-?\d+\.?\d*(deg|rad|grad|turn)$/gi.test(this.authored);
},
get specialValue() {
return SPECIALVALUES.has(this.lowerCased) ? this.authored : null;
},
get deg() {
let invalidOrSpecialValue = this._getInvalidOrSpecialValue();
if (invalidOrSpecialValue !== false) {
return invalidOrSpecialValue;
}
let angleUnit = classifyAngle(this.authored);
if (angleUnit === CssAngle.ANGLEUNIT.deg) {
// The angle is valid and is in degree.
return this.authored;
}
let degValue;
if (angleUnit === CssAngle.ANGLEUNIT.rad) {
// The angle is valid and is in radian.
degValue = this.authoredAngleValue / (Math.PI / 180);
}
if (angleUnit === CssAngle.ANGLEUNIT.grad) {
// The angle is valid and is in gradian.
degValue = this.authoredAngleValue * 0.9;
}
if (angleUnit === CssAngle.ANGLEUNIT.turn) {
// The angle is valid and is in turn.
degValue = this.authoredAngleValue * 360;
}
let unitStr = CssAngle.ANGLEUNIT.deg;
if (this._angleUnitUppercase === true) {
unitStr = unitStr.toUpperCase();
}
return `${Math.round(degValue * 100) / 100}${unitStr}`;
},
get rad() {
let invalidOrSpecialValue = this._getInvalidOrSpecialValue();
if (invalidOrSpecialValue !== false) {
return invalidOrSpecialValue;
}
let unit = classifyAngle(this.authored);
if (unit === CssAngle.ANGLEUNIT.rad) {
// The angle is valid and is in radian.
return this.authored;
}
let radValue;
if (unit === CssAngle.ANGLEUNIT.deg) {
// The angle is valid and is in degree.
radValue = this.authoredAngleValue * (Math.PI / 180);
}
if (unit === CssAngle.ANGLEUNIT.grad) {
// The angle is valid and is in gradian.
radValue = this.authoredAngleValue * 0.9 * (Math.PI / 180);
}
if (unit === CssAngle.ANGLEUNIT.turn) {
// The angle is valid and is in turn.
radValue = this.authoredAngleValue * 360 * (Math.PI / 180);
}
let unitStr = CssAngle.ANGLEUNIT.rad;
if (this._angleUnitUppercase === true) {
unitStr = unitStr.toUpperCase();
}
return `${Math.round(radValue * 10000) / 10000}${unitStr}`;
},
get grad() {
let invalidOrSpecialValue = this._getInvalidOrSpecialValue();
if (invalidOrSpecialValue !== false) {
return invalidOrSpecialValue;
}
let unit = classifyAngle(this.authored);
if (unit === CssAngle.ANGLEUNIT.grad) {
// The angle is valid and is in gradian
return this.authored;
}
let gradValue;
if (unit === CssAngle.ANGLEUNIT.deg) {
// The angle is valid and is in degree
gradValue = this.authoredAngleValue / 0.9;
}
if (unit === CssAngle.ANGLEUNIT.rad) {
// The angle is valid and is in radian
gradValue = this.authoredAngleValue / 0.9 / (Math.PI / 180);
}
if (unit === CssAngle.ANGLEUNIT.turn) {
// The angle is valid and is in turn
gradValue = this.authoredAngleValue * 400;
}
let unitStr = CssAngle.ANGLEUNIT.grad;
if (this._angleUnitUppercase === true) {
unitStr = unitStr.toUpperCase();
}
return `${Math.round(gradValue * 100) / 100}${unitStr}`;
},
get turn() {
let invalidOrSpecialValue = this._getInvalidOrSpecialValue();
if (invalidOrSpecialValue !== false) {
return invalidOrSpecialValue;
}
let unit = classifyAngle(this.authored);
if (unit === CssAngle.ANGLEUNIT.turn) {
// The angle is valid and is in turn
return this.authored;
}
let turnValue;
if (unit === CssAngle.ANGLEUNIT.deg) {
// The angle is valid and is in degree
turnValue = this.authoredAngleValue / 360;
}
if (unit === CssAngle.ANGLEUNIT.rad) {
// The angle is valid and is in radian
turnValue = (this.authoredAngleValue / (Math.PI / 180)) / 360;
}
if (unit === CssAngle.ANGLEUNIT.grad) {
// The angle is valid and is in gradian
turnValue = this.authoredAngleValue / 400;
}
let unitStr = CssAngle.ANGLEUNIT.turn;
if (this._angleUnitUppercase === true) {
unitStr = unitStr.toUpperCase();
}
return `${Math.round(turnValue * 100) / 100}${unitStr}`;
},
/**
* Check whether the angle value is in the special list e.g.
* inherit or invalid.
*
* @return {String|Boolean}
* - If the current angle is a special value e.g. "inherit" then
* return the angle.
* - If the angle is invalid return an empty string.
* - If the angle is a regular angle e.g. 90deg so we return false
* to indicate that the angle is neither invalid nor special.
*/
_getInvalidOrSpecialValue: function() {
if (this.specialValue) {
return this.specialValue;
}
if (!this.valid) {
return "";
}
return false;
},
/**
* Change angle
*
* @param {String} angle
* Any valid angle value + unit string
*/
newAngle: function(angle) {
// Store a lower-cased version of the angle to help with format
// testing. The original text is kept as well so it can be
// returned when needed.
this.lowerCased = angle.toLowerCase();
this._angleUnitUppercase = (angle === angle.toUpperCase());
this.authored = angle;
let reg = new RegExp(
`(${Object.keys(CssAngle.ANGLEUNIT).join("|")})$`, "i");
let unitStartIdx = angle.search(reg);
this.authoredAngleValue = angle.substring(0, unitStartIdx);
this.authoredAngleUnit = angle.substring(unitStartIdx, angle.length);
return this;
},
nextAngleUnit: function() {
// Get a reordered array from the formats object
// to have the current format at the front so we can cycle through.
let formats = Object.keys(CssAngle.ANGLEUNIT);
let putOnEnd = formats.splice(0, formats.indexOf(this.angleUnit));
formats = formats.concat(putOnEnd);
let currentDisplayedValue = this[formats[0]];
for (let format of formats) {
if (this[format].toLowerCase() !== currentDisplayedValue.toLowerCase()) {
this.angleUnit = CssAngle.ANGLEUNIT[format];
break;
}
}
return this.toString();
},
/**
* Return a string representing a angle
*/
toString: function() {
let angle;
switch (this.angleUnit) {
case CssAngle.ANGLEUNIT.deg:
angle = this.deg;
break;
case CssAngle.ANGLEUNIT.rad:
angle = this.rad;
break;
case CssAngle.ANGLEUNIT.grad:
angle = this.grad;
break;
case CssAngle.ANGLEUNIT.turn:
angle = this.turn;
break;
default:
angle = this.deg;
}
if (this._angleUnitUppercase &&
this.angleUnit != CssAngle.ANGLEUNIT.authored) {
angle = angle.toUpperCase();
}
return angle;
},
/**
* This method allows comparison of CssAngle objects using ===.
*/
valueOf: function() {
return this.deg;
},
};
/**
* Given a color, classify its type as one of the possible angle
* units, as known by |CssAngle.angleUnit|.
*
* @param {String} value
* The angle, in any form accepted by CSS.
* @return {String}
* The angle classification, one of "deg", "rad", "grad", or "turn".
*/
function classifyAngle(value) {
value = value.toLowerCase();
if (value.endsWith("deg")) {
return CssAngle.ANGLEUNIT.deg;
}
if (value.endsWith("grad")) {
return CssAngle.ANGLEUNIT.grad;
}
if (value.endsWith("rad")) {
return CssAngle.ANGLEUNIT.rad;
}
if (value.endsWith("turn")) {
return CssAngle.ANGLEUNIT.turn;
}
return CssAngle.ANGLEUNIT.deg;
}