diff --git a/cypress/e2e/astral-accessibility.cy.ts b/cypress/e2e/astral-accessibility.cy.ts
index b9ff0ac..cbb1e76 100644
--- a/cypress/e2e/astral-accessibility.cy.ts
+++ b/cypress/e2e/astral-accessibility.cy.ts
@@ -190,5 +190,20 @@ describe('template spec', () => {
cy.wrap(elmt).should('have.css', 'font-size', `${initialSize}px`);
}
});
+
+ const screenMaskComponent = cy.document().find('astral-screen-mask');
+ // cursor size check
+ cy.document().find('style').should('not.contain', "cursor: url(''), auto;");
+ screenMaskComponent.click();
+ cy.document().find('style').should('contain', "cursor: url(''), auto;");
+
+ // screen mask check
+
+ cy.document().find('.mask-top').should('not.exist')
+ screenMaskComponent.click();
+ cy.document().find('.mask-top').should('exist')
+ cy.reload()
+ cy.document().find('.mask-top').should('not.exist')
+
});
});
diff --git a/projects/astral-accessibility/src/lib/astral-accessibility.component.html b/projects/astral-accessibility/src/lib/astral-accessibility.component.html
index 3874397..a3ba16f 100644
--- a/projects/astral-accessibility/src/lib/astral-accessibility.component.html
+++ b/projects/astral-accessibility/src/lib/astral-accessibility.component.html
@@ -45,6 +45,12 @@
+
+
diff --git a/projects/astral-accessibility/src/lib/astral-accessibility.component.scss b/projects/astral-accessibility/src/lib/astral-accessibility.component.scss
index 55b78c4..27656bc 100644
--- a/projects/astral-accessibility/src/lib/astral-accessibility.component.scss
+++ b/projects/astral-accessibility/src/lib/astral-accessibility.component.scss
@@ -282,4 +282,4 @@
word-spacing: 0.48em;
letter-spacing: 0.36em;
}
-}
+}
\ No newline at end of file
diff --git a/projects/astral-accessibility/src/lib/astral-accessibility.component.ts b/projects/astral-accessibility/src/lib/astral-accessibility.component.ts
index c9f50c4..1d32608 100644
--- a/projects/astral-accessibility/src/lib/astral-accessibility.component.ts
+++ b/projects/astral-accessibility/src/lib/astral-accessibility.component.ts
@@ -6,6 +6,8 @@ import { SaturateComponent } from './controls/saturate.component';
import { TextSizeComponent } from './controls/text-size.component';
import { TextSpacingComponent } from './controls/text-spacing.component';
import { ScreenReaderComponent } from './controls/screen-reader.component';
+import { ScreenMaskComponent } from './controls/screen-mask.component';
+import { LineHeightComponent } from './controls/line-height.component';
@Component({
selector: 'astral-accessibility',
@@ -20,6 +22,8 @@ import { ScreenReaderComponent } from './controls/screen-reader.component';
TextSizeComponent,
TextSpacingComponent,
ScreenReaderComponent,
+ ScreenMaskComponent,
+ LineHeightComponent
],
schemas: [CUSTOM_ELEMENTS_SCHEMA]
})
diff --git a/projects/astral-accessibility/src/lib/controls/line-height.component.ts b/projects/astral-accessibility/src/lib/controls/line-height.component.ts
new file mode 100644
index 0000000..a486ff6
--- /dev/null
+++ b/projects/astral-accessibility/src/lib/controls/line-height.component.ts
@@ -0,0 +1,140 @@
+import { DOCUMENT, NgIf, NgClass } from '@angular/common';
+import { Component, Renderer2, inject } from '@angular/core';
+import { AstralCheckmarkSvgComponent } from '../util/astral-checksvg.component';
+
+@Component({
+ selector: 'astral-line-height',
+ standalone: true,
+ template: `
+
+
+
+
+
+
+
{{ states[currentState] }}
+
+
+
+
+
+
+
+ `,
+ imports: [NgIf, NgClass, AstralCheckmarkSvgComponent],
+})
+export class LineHeightComponent {
+ constructor(private renderer: Renderer2) {}
+ document = inject(DOCUMENT);
+
+ currentState = 0;
+ base = 'Line Height';
+ states = [this.base, 'Light Height', 'Moderate Height', 'Heavy Height'];
+
+ lowHeight = `
+ *{
+ line-height: 1.5 !important;
+ }`;
+
+ moderateHeight = `
+ *{
+ line-height: 3 !important;
+ }`;
+
+ heavyHeight = `
+ *{
+ line-height: 4 !important;
+ }`;
+
+ private lowHeightStyleTag: HTMLStyleElement | null = null;
+ private moderateHeightStyleTag: HTMLStyleElement | null = null;
+ private heavyHeightStyleTag: HTMLStyleElement | null = null;
+
+ _style: HTMLStyleElement;
+
+ nextState() {
+ this.currentState += 1;
+ this.currentState = this.currentState % 4;
+
+ this._runStateLogic();
+ }
+
+ private _runStateLogic() {
+ this._style?.remove?.();
+ this._style = this.document.createElement('style');
+
+ if (this.states[this.currentState] === 'Light Height') {
+ if (!this.lowHeightStyleTag) {
+ this.lowHeightStyleTag = this.renderer.createElement('style');
+ this.renderer.appendChild(this.lowHeightStyleTag, this.renderer.createText(this.lowHeight));
+ this.renderer.appendChild(this.document.head, this.lowHeightStyleTag);
+ }
+ } else {
+ if (this.lowHeightStyleTag) {
+ this.renderer.removeChild(this.document.head, this.lowHeightStyleTag);
+ this.lowHeightStyleTag = null;
+ }
+ }
+
+ if (this.states[this.currentState] === 'Moderate Height') {
+ if (!this.moderateHeightStyleTag) {
+ this.moderateHeightStyleTag = this.renderer.createElement('style');
+ this.renderer.appendChild(this.moderateHeightStyleTag, this.renderer.createText(this.moderateHeight));
+ this.renderer.appendChild(this.document.head, this.moderateHeightStyleTag);
+ }
+ } else {
+ if (this.moderateHeightStyleTag) {
+ this.renderer.removeChild(this.document.head, this.moderateHeightStyleTag);
+ this.moderateHeightStyleTag = null;
+ }
+ }
+
+ if (this.states[this.currentState] === 'Heavy Height') {
+ if (!this.heavyHeightStyleTag) {
+ this.heavyHeightStyleTag = this.renderer.createElement('style');
+ this.renderer.appendChild(this.heavyHeightStyleTag, this.renderer.createText(this.heavyHeight));
+ this.renderer.appendChild(this.document.head, this.heavyHeightStyleTag);
+ }
+ } else {
+ if (this.heavyHeightStyleTag) {
+ this.renderer.removeChild(this.document.head, this.heavyHeightStyleTag);
+ this.heavyHeightStyleTag = null;
+ }
+ }
+
+ this.document.body.appendChild(this._style);
+ }
+}
diff --git a/projects/astral-accessibility/src/lib/controls/screen-mask.component.ts b/projects/astral-accessibility/src/lib/controls/screen-mask.component.ts
new file mode 100644
index 0000000..65a50c2
--- /dev/null
+++ b/projects/astral-accessibility/src/lib/controls/screen-mask.component.ts
@@ -0,0 +1,229 @@
+import { DOCUMENT, NgIf, NgClass } from '@angular/common';
+import { Component, Renderer2, inject } from '@angular/core';
+import { AstralCheckmarkSvgComponent } from '../util/astral-checksvg.component';
+
+@Component({
+ selector: 'astral-screen-mask',
+ standalone: true,
+ template: `
+
+
+
+
+
+
+
{{ states[currentState] }}
+
+
+
+
+
+
+
+
+
+ `,
+ imports: [NgIf, NgClass, AstralCheckmarkSvgComponent],
+})
+export class ScreenMaskComponent {
+ constructor(private renderer: Renderer2) {}
+
+ cursorY = 0;
+ screenHeight: number = window.innerHeight;
+ height = this.screenHeight - this.cursorY;
+ listenersActive: boolean = false;
+
+ screenMaskContainerStyle = `
+ .screen-mask-container {
+ position: fixed;
+ top: 0;
+ left: 0;
+ width: 100%;
+ height: 100%;
+ pointer-events: none;
+ z-index: 9999;
+ }
+ `;
+
+ maskTopStyle = `
+ .mask-top {
+ position: absolute;
+ top: 0;
+ width: 100%;
+ background-color: rgba(0, 0, 0, 0.55);
+ pointer-events: none;
+ border-bottom: 8px solid #654AFF;
+ height: ${this.height - 115}px;
+ }
+ `;
+
+ maskBottomStyle = `
+ .mask-bottom {
+ position: absolute;
+ bottom: 0;
+ width: 100%;
+ background-color: rgba(0, 0, 0, 0.55);
+ pointer-events: none;
+ border-top: 8px solid #FFCB00;
+ height: ${this.cursorY}px;
+ }
+ `;
+
+ toggleListeners(active: boolean) {
+ this.listenersActive = active;
+
+ if (this.listenersActive) {
+ this.addEventListeners();
+ this.updateMaskStyles();
+ } else {
+ this.removeEventListeners();
+ }
+ }
+
+ // Method to add the event listeners
+ private addEventListeners() {
+ window.addEventListener('mousemove', this.onMouseMove);
+ window.addEventListener('resize', this.onResize);
+ }
+
+ // Method to remove the event listeners
+ private removeEventListeners() {
+ window.removeEventListener('mousemove', this.onMouseMove);
+ window.removeEventListener('resize', this.onResize);
+ }
+
+ private onMouseMove = (event: MouseEvent) => {
+ if (!this.listenersActive) return; // Don't proceed if listeners are paused
+
+ this.cursorY = event.clientY;
+ this.height = this.screenHeight - this.cursorY;
+ this.updateMaskStyles();
+ }
+
+ // Event listener method for resize
+ private onResize = (event: any) => {
+ if (!this.listenersActive) return; // Don't proceed if listeners are paused
+
+ this.screenHeight = event.target.innerHeight;
+ this.height = this.screenHeight - this.cursorY;
+ this.updateMaskStyles();
+ }
+
+ updateMaskStyles() {
+ const maskTop = document.querySelector('.mask-top');
+ const maskBottom = document.querySelector('.mask-bottom');
+
+ if (maskTop) {
+ this.renderer.setStyle(maskTop, 'height', `${this.cursorY - 57}px`);
+ }
+ if (maskBottom) {
+ this.renderer.setStyle(maskBottom, 'height', `${this.height - 57}px`);
+ }
+ }
+
+ document = inject(DOCUMENT);
+
+ currentState = 0;
+ base = 'Screen Mask';
+ states = [this.base, 'Large Cursor', 'Reading Mask'];
+
+ _style: HTMLStyleElement;
+
+ nextState() {
+ this.currentState += 1;
+ this.currentState = this.currentState % 3;
+
+ this._runStateLogic();
+ }
+
+ private _runStateLogic() {
+ this._style?.remove?.();
+ this._style = this.document.createElement('style');
+
+ if (this.states[this.currentState] === 'Large Cursor') {
+ this._style.textContent = `
+ body, *{
+ cursor: url(''), auto;
+ }
+ `;
+ }
+
+ if (this.states[this.currentState] === 'Reading Mask') {
+
+ const screenMaskContainer = this.renderer.createElement('div');
+ const maskTop = this.renderer.createElement('div');
+ const maskBottom = this.renderer.createElement('div');
+
+ this.renderer.addClass(screenMaskContainer, 'screen-mask-container');
+ this.renderer.addClass(maskTop, 'mask-top');
+ this.renderer.addClass(maskBottom, 'mask-bottom');
+
+ const maskBottomCss = this.renderer.createElement('style');
+ this.renderer.appendChild(maskBottomCss, this.renderer.createText(this.maskBottomStyle));
+ this.renderer.appendChild(this.document.head, maskBottomCss);
+
+ const maskTopCss = this.renderer.createElement('style');
+ this.renderer.appendChild(maskTopCss, this.renderer.createText(this.maskTopStyle));
+ this.renderer.appendChild(this.document.head, maskTopCss);
+
+ const screenMaskContainerCss = this.renderer.createElement('style');
+ this.renderer.appendChild(screenMaskContainerCss, this.renderer.createText(this.screenMaskContainerStyle));
+ this.renderer.appendChild(this.document.head, screenMaskContainerCss);
+
+ screenMaskContainer.appendChild(maskTop);
+ screenMaskContainer.appendChild(maskBottom);
+ this.document.body.appendChild(screenMaskContainer);
+
+ this.toggleListeners(true);
+
+ } else {
+ const removeMaskContainer = document.querySelector('.screen-mask-container');
+ if (removeMaskContainer) {
+ this.renderer.removeChild(document.body, removeMaskContainer);
+ }
+ this.toggleListeners(false);
+ }
+
+ this.document.body.appendChild(this._style);
+ }
+}
\ No newline at end of file
diff --git a/projects/demo/index.html b/projects/demo/index.html
index 1ad6529..704c267 100644
--- a/projects/demo/index.html
+++ b/projects/demo/index.html
@@ -22,8 +22,49 @@ Welcome to Astral Accessibility!
reading accessibility functionalities like text size, text line
height/spacing, text reader, contrast, invert, saturation. It is
located at the bottom right corner of the application.
-
- Screen Reader
+
+
Screen reader is a tool where it would reads out texts on screen
where user clicks on. For any html elements, if an aria label is
@@ -31,29 +72,240 @@
Screen Reader
otherwise, it reads the text content of the element. There are 3
different speeds, normal, fast and slow.
- Contrast
+
+
Contrast is a tool that removes background and replaces it with
black or white to increase the difference in colours between text
and the background to increase legibility. There are 3 modes, the
invert colours, high contrast, and dark high contrast.
- Saturation
+
+
Saturation is a tool that adjusts how colourful the colours are on
screen, it has 3 different modes to lower saturation, increase
saturation, or remove all the colours on screen (black and white).
- Bigger Text
+
+
Bigger Text is a tool that increases the size of the texts on
screen.
- Text Spacing
+
+
Text Spacing is a tool that increases the spacing between each
character on the screen to increase legibility and readibility.
+
+
+
+ Screen Mask is a tool which dims the background and has a horizontal
+ focus area which follows the cursor sliding horizontally.
+
+
+
+
+ Line height is a tool which increased the line height to increase readibility.
+
Check box demo