From 5967ab45ae8445c4a57c56613cecd549a54b6e1b Mon Sep 17 00:00:00 2001 From: yinm Date: Fri, 13 Sep 2024 00:04:04 +0900 Subject: [PATCH 1/4] feat: [#1147] Adds support for the aspect-ratio property --- .../css/declaration/CSSStyleDeclaration.ts | 8 +++++ .../CSSStyleDeclarationPropertyManager.ts | 3 ++ .../CSSStyleDeclarationPropertySetParser.ts | 34 +++++++++++++++++++ 3 files changed, 45 insertions(+) diff --git a/packages/happy-dom/src/css/declaration/CSSStyleDeclaration.ts b/packages/happy-dom/src/css/declaration/CSSStyleDeclaration.ts index c394d72bf..1bd302f49 100644 --- a/packages/happy-dom/src/css/declaration/CSSStyleDeclaration.ts +++ b/packages/happy-dom/src/css/declaration/CSSStyleDeclaration.ts @@ -4740,4 +4740,12 @@ export default class CSSStyleDeclaration extends AbstractCSSStyleDeclaration { public set containerName(value: string) { this.setProperty('container-name', value); } + + public get aspectRatio(): string { + return this.getPropertyValue('aspect-ratio'); + } + + public set aspectRatio(value: string) { + this.setProperty('aspect-ratio', value); + } } diff --git a/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertyManager.ts b/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertyManager.ts index ce1a969c8..bab47d96e 100644 --- a/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertyManager.ts +++ b/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertyManager.ts @@ -509,6 +509,9 @@ export default class CSSStyleDeclarationPropertyManager { case 'visibility': properties = CSSStyleDeclarationPropertySetParser.getVisibility(value, important); break; + case 'aspect-ratio': + properties = CSSStyleDeclarationPropertySetParser.getAspectRatio(value, important); + break; default: const trimmedValue = value.trim(); diff --git a/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts b/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts index 39d9f7b04..8911f8a25 100644 --- a/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts +++ b/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts @@ -3256,4 +3256,38 @@ export default class CSSStyleDeclarationPropertySetParser { } return null; } + + /** + * Returns aspect ratio. + * + * @param value Value. + * @param important Important. + * @returns Property + */ + public static getAspectRatio( + value: string, + important: boolean + ): { + [key: string]: ICSSStyleDeclarationPropertyValue; + } { + const variable = CSSStyleDeclarationValueParser.getVariable(value); + if (variable) { + return { 'aspect-ratio': { value: variable, important } }; + } + + const lowerValue = value.toLowerCase(); + if (CSSStyleDeclarationValueParser.getGlobal(lowerValue)) { + return { 'aspect-ratio': { value: lowerValue, important } }; + } + + const parts = value.split('/'); + if (parts.length === 1 && !Number.isNaN(Number(parts[0]))) { + return { 'aspect-ratio': { value, important } }; + } + if (parts.length === 2 && !Number.isNaN(Number(parts[0])) && !Number.isNaN(Number(parts[1]))) { + return { 'aspect-ratio': { value, important } }; + } + + return null; + } } From 60a7860150b8f120830bc858eab78c68237e19c4 Mon Sep 17 00:00:00 2001 From: David Ortner Date: Tue, 5 Nov 2024 01:39:42 +0100 Subject: [PATCH 2/4] chore: [#1147] Adds support for aspect-ratio to CSSStyleDeclaration --- .../CSSStyleDeclarationPropertySetParser.ts | 113 +++++++++++------- .../declaration/CSSStyleDeclaration.test.ts | 38 ++++++ 2 files changed, 109 insertions(+), 42 deletions(-) diff --git a/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts b/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts index 8911f8a25..79957277c 100644 --- a/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts +++ b/packages/happy-dom/src/css/declaration/property-manager/CSSStyleDeclarationPropertySetParser.ts @@ -2,8 +2,9 @@ import CSSStyleDeclarationValueParser from './CSSStyleDeclarationValueParser.js' import ICSSStyleDeclarationPropertyValue from './ICSSStyleDeclarationPropertyValue.js'; const RECT_REGEXP = /^rect\((.*)\)$/i; -const SPLIT_COMMA_SEPARATED_REGEXP = /,(?=(?:(?:(?!\))[\s\S])*\()|[^\(\)]*$)/; // Split on commas that are outside of parentheses -const SPLIT_SPACE_SEPARATED_REGEXP = /\s+(?=(?:(?:(?!\))[\s\S])*\()|[^\(\)]*$)/; // Split on spaces that are outside of parentheses +const SPLIT_COMMA_SEPARATED_WITH_PARANTHESES_REGEXP = /,(?=(?:(?:(?!\))[\s\S])*\()|[^\(\)]*$)/; // Split on commas that are outside of parentheses +const SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP = /\s+(?=(?:(?:(?!\))[\s\S])*\()|[^\(\)]*$)/; // Split on spaces that are outside of parentheses +const WHITE_SPACE_GLOBAL_REGEXP = /\s+/gm; const BORDER_STYLE = [ 'none', 'hidden', @@ -497,7 +498,7 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getOutlineWidth('initial', important) }; - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (const part of parts) { const width = this.getOutlineWidth(part, important); @@ -534,7 +535,7 @@ export default class CSSStyleDeclarationPropertySetParser { return color ? { 'outline-color': { value: color, important } - } + } : null; } @@ -649,7 +650,9 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getBorderImage('initial', important) }; - const parts = value.replace(/\s*,\s*/g, ',').split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value + .replace(/\s*,\s*/g, ',') + .split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (const part of parts) { const width = this.getBorderWidth(part, important); @@ -695,7 +698,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const top = this.getBorderTopWidth(parts[0], important); const right = this.getBorderRightWidth(parts[1] || parts[0], important); const bottom = this.getBorderBottomWidth(parts[2] || parts[0], important); @@ -741,7 +744,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const top = this.getBorderTopStyle(parts[0], important); const right = this.getBorderRightStyle(parts[1] || parts[0], important); const bottom = this.getBorderBottomStyle(parts[2] || parts[0], important); @@ -788,7 +791,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const top = this.getBorderTopColor(parts[0], important); const right = this.getBorderRightColor(parts[1] || parts[0], important); const bottom = this.getBorderBottomColor(parts[2] || parts[0], important); @@ -843,7 +846,7 @@ export default class CSSStyleDeclarationPropertySetParser { parsedValue = parsedValue.replace(sourceMatch[0], ''); } - const parts = parsedValue.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = parsedValue.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (sourceMatch) { parts.push(sourceMatch[1]); @@ -1038,7 +1041,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = lowerValue.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = lowerValue.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (parts.length > 4) { return null; @@ -1099,7 +1102,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (parts.length > 4) { return null; @@ -1154,7 +1157,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = lowerValue.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = lowerValue.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (parts.length > 2) { return null; @@ -1430,7 +1433,7 @@ export default class CSSStyleDeclarationPropertySetParser { return color ? { 'border-top-color': { value: color, important } - } + } : null; } @@ -1458,7 +1461,7 @@ export default class CSSStyleDeclarationPropertySetParser { return color ? { 'border-right-color': { value: color, important } - } + } : null; } @@ -1486,7 +1489,7 @@ export default class CSSStyleDeclarationPropertySetParser { return color ? { 'border-bottom-color': { value: color, important } - } + } : null; } @@ -1514,7 +1517,7 @@ export default class CSSStyleDeclarationPropertySetParser { return color ? { 'border-left-color': { value: color, important } - } + } : null; } @@ -1545,7 +1548,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const topLeft = this.getBorderTopLeftRadius(parts[0], important); const topRight = this.getBorderTopRightRadius(parts[1] || parts[0], important); const bottomRight = this.getBorderBottomRightRadius(parts[2] || parts[0], important); @@ -1683,7 +1686,7 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getBorderTopColor('initial', important) }; - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (const part of parts) { const width = this.getBorderTopWidth(part, important); @@ -1732,7 +1735,7 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getBorderRightColor('initial', important) }; - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (const part of parts) { const width = this.getBorderRightWidth(part, important); @@ -1781,7 +1784,7 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getBorderBottomColor('initial', important) }; - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (const part of parts) { const width = this.getBorderBottomWidth(part, important); @@ -1830,7 +1833,7 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getBorderLeftColor('initial', important) }; - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (const part of parts) { const width = this.getBorderLeftWidth(part, important); @@ -1873,7 +1876,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const top = this.getPaddingTop(parts[0], important); const right = this.getPaddingRight(parts[1] || parts[0], important); const bottom = this.getPaddingBottom(parts[2] || parts[0], important); @@ -2006,7 +2009,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const top = this.getMarginTop(parts[0], important); const right = this.getMarginRight(parts[1] || parts[0], important); const bottom = this.getMarginBottom(parts[2] || parts[0], important); @@ -2164,7 +2167,7 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const flexGrow = this.getFlexGrow(parts[0], important); const flexShrink = this.getFlexShrink(parts[1] || '1', important); const flexBasis = this.getFlexBasis(parts[2] || '0%', important); @@ -2300,7 +2303,7 @@ export default class CSSStyleDeclarationPropertySetParser { const parts = value .replace(/\s,\s/g, ',') .replace(/\s\/\s/g, '/') - .split(SPLIT_SPACE_SEPARATED_REGEXP); + .split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); const backgroundPositions = []; @@ -2397,7 +2400,7 @@ export default class CSSStyleDeclarationPropertySetParser { return { 'background-size': { value: lowerValue, important } }; } - const imageParts = lowerValue.split(SPLIT_COMMA_SEPARATED_REGEXP); + const imageParts = lowerValue.split(SPLIT_COMMA_SEPARATED_WITH_PARANTHESES_REGEXP); const parsed = []; for (const imagePart of imageParts) { @@ -2568,12 +2571,12 @@ export default class CSSStyleDeclarationPropertySetParser { }; } - const imageParts = value.split(SPLIT_COMMA_SEPARATED_REGEXP); + const imageParts = value.split(SPLIT_COMMA_SEPARATED_WITH_PARANTHESES_REGEXP); let x = ''; let y = ''; for (const imagePart of imageParts) { - const parts = imagePart.trim().split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = imagePart.trim().split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (x) { x += ','; @@ -2681,11 +2684,11 @@ export default class CSSStyleDeclarationPropertySetParser { return { 'background-position-x': { value: lowerValue, important } }; } - const imageParts = lowerValue.split(SPLIT_COMMA_SEPARATED_REGEXP); + const imageParts = lowerValue.split(SPLIT_COMMA_SEPARATED_WITH_PARANTHESES_REGEXP); let parsedValue = ''; for (const imagePart of imageParts) { - const parts = imagePart.trim().split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = imagePart.trim().split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (parsedValue) { parsedValue += ','; @@ -2732,11 +2735,11 @@ export default class CSSStyleDeclarationPropertySetParser { return { 'background-position-y': { value: lowerValue, important } }; } - const imageParts = lowerValue.split(SPLIT_COMMA_SEPARATED_REGEXP); + const imageParts = lowerValue.split(SPLIT_COMMA_SEPARATED_WITH_PARANTHESES_REGEXP); let parsedValue = ''; for (const imagePart of imageParts) { - const parts = imagePart.trim().split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = imagePart.trim().split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (parsedValue) { parsedValue += ','; @@ -2782,7 +2785,7 @@ export default class CSSStyleDeclarationPropertySetParser { return color ? { ['background-color']: { important, value: color } - } + } : null; } @@ -2808,7 +2811,7 @@ export default class CSSStyleDeclarationPropertySetParser { return { 'background-image': { value: lowerValue, important } }; } - const parts = value.split(SPLIT_COMMA_SEPARATED_REGEXP); + const parts = value.split(SPLIT_COMMA_SEPARATED_WITH_PARANTHESES_REGEXP); const parsed = []; for (const part of parts) { @@ -2917,7 +2920,9 @@ export default class CSSStyleDeclarationPropertySetParser { ...this.getLineHeight('normal', important) }; - const parts = value.replace(/\s*\/\s*/g, '/').split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value + .replace(/\s*\/\s*/g, '/') + .split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); for (let i = 0, max = parts.length; i < max; i++) { const part = parts[i]; @@ -2985,7 +2990,7 @@ export default class CSSStyleDeclarationPropertySetParser { if (CSSStyleDeclarationValueParser.getGlobal(lowerValue) || FONT_STYLE.includes(lowerValue)) { return { 'font-style': { value: lowerValue, important } }; } - const parts = value.split(SPLIT_SPACE_SEPARATED_REGEXP); + const parts = value.split(SPLIT_SPACE_SEPARATED_WITH_PARANTHESES_REGEXP); if (parts.length === 2 && parts[0] === 'oblique') { const degree = CSSStyleDeclarationValueParser.getDegree(parts[1]); return degree ? { 'font-style': { value: lowerValue, important } } : null; @@ -3276,18 +3281,42 @@ export default class CSSStyleDeclarationPropertySetParser { } const lowerValue = value.toLowerCase(); + if (CSSStyleDeclarationValueParser.getGlobal(lowerValue)) { return { 'aspect-ratio': { value: lowerValue, important } }; } - const parts = value.split('/'); - if (parts.length === 1 && !Number.isNaN(Number(parts[0]))) { - return { 'aspect-ratio': { value, important } }; + let parsedValue = value; + + const hasAuto = parsedValue.includes('auto'); + + if (hasAuto) { + parsedValue = parsedValue.replace('auto', ''); } - if (parts.length === 2 && !Number.isNaN(Number(parts[0])) && !Number.isNaN(Number(parts[1]))) { - return { 'aspect-ratio': { value, important } }; + + parsedValue = parsedValue.replace(WHITE_SPACE_GLOBAL_REGEXP, ''); + + if (!parsedValue) { + return { 'aspect-ratio': { value: 'auto', important } }; } - return null; + const aspectRatio = parsedValue.split('/'); + + if (aspectRatio.length > 3) { + return null; + } + + const width = Number(aspectRatio[0]); + const height = aspectRatio[1] ? Number(aspectRatio[1]) : 1; + + if (isNaN(width) || isNaN(height)) { + return null; + } + + if (hasAuto) { + return { 'aspect-ratio': { value: `auto ${width} / ${height}`, important } }; + } + + return { 'aspect-ratio': { value: `${width} / ${height}`, important } }; } } diff --git a/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts b/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts index 2af6c4cb2..7a3b97d57 100644 --- a/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts +++ b/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts @@ -2553,6 +2553,44 @@ describe('CSSStyleDeclaration', () => { }); }); + describe('get aspectRatio()', () => { + it('Returns style property.', () => { + const declaration = new CSSStyleDeclaration(element); + + for (const value of [ + 'var(--test-variable)', + 'inherit', + 'initial', + 'revert', + 'unset', + 'auto', + '1 / 1', + '16 / 9', + '4 / 3', + '1 / 2', + '2 / 1', + '3 / 4', + '9 / 16' + ]) { + element.setAttribute('style', `aspect-ratio: ${value}`); + + expect(declaration.aspectRatio).toBe(value); + } + + element.setAttribute('style', 'aspect-ratio: 2'); + + expect(declaration.aspectRatio).toBe('2 / 1'); + + element.setAttribute('style', 'aspect-ratio: 16/9 auto'); + + expect(declaration.aspectRatio).toBe('auto 16 / 9'); + + element.setAttribute('style', 'aspect-ratio: 16/9'); + + expect(declaration.aspectRatio).toBe('16 / 9'); + }); + }); + describe('get length()', () => { it('Returns length when of styles on element.', () => { const declaration = new CSSStyleDeclaration(element); From 814c7c48271e2cbb8f8b913c63ce7274723db6c1 Mon Sep 17 00:00:00 2001 From: David Ortner Date: Tue, 5 Nov 2024 01:50:43 +0100 Subject: [PATCH 3/4] chore: [#1147] Adds support for aspect-ratio to CSSStyleDeclaration --- .../happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts b/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts index 054c1c2f6..2f13cf082 100644 --- a/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts +++ b/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts @@ -2730,7 +2730,7 @@ describe('CSSStyleDeclaration', () => { describe('get aspectRatio()', () => { it('Returns style property.', () => { - const declaration = new CSSStyleDeclaration(element); + const declaration = new CSSStyleDeclaration(PropertySymbol.illegalConstructor, element); for (const value of [ 'var(--test-variable)', From f2c1f8df0f1cb720e83666fadc7a7ae102da7f43 Mon Sep 17 00:00:00 2001 From: David Ortner Date: Tue, 5 Nov 2024 01:51:11 +0100 Subject: [PATCH 4/4] chore: [#1147] Adds support for aspect-ratio to CSSStyleDeclaration --- .../test/css/declaration/CSSStyleDeclaration.test.ts | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts b/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts index 2f13cf082..757885e78 100644 --- a/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts +++ b/packages/happy-dom/test/css/declaration/CSSStyleDeclaration.test.ts @@ -2730,7 +2730,9 @@ describe('CSSStyleDeclaration', () => { describe('get aspectRatio()', () => { it('Returns style property.', () => { - const declaration = new CSSStyleDeclaration(PropertySymbol.illegalConstructor, element); + const declaration = new CSSStyleDeclaration(PropertySymbol.illegalConstructor, window, { + element + }); for (const value of [ 'var(--test-variable)',