src/components/sq-overlay/sq-overlay.component.ts
Represents an overlay component, an abstraction with differente style but still a modal.
Example :<sq-overlay [open]="isOverlayOpen" overlayDirection="right" (overlayClose)="onOverlayClose()">
<ng-template #headerTemplate>
<h2>Overlay Header</h2>
</ng-template>
<ng-template #footerTemplate>
Footer
</ng-template>
<div>
<!-- Your content here -->
</div>
</sq-overlay>
<button (click)='isOverlayOpen = true'>Open Modal</button>
OnChanges
OnDestroy
selector | sq-overlay |
styleUrls | ./sq-overlay.component.scss |
templateUrl | ./sq-overlay.component.html |
Properties |
|
Methods |
Inputs |
Outputs |
constructor(documentImported: Document, router: Router, getWindow: GetWindow)
|
||||||||||||||||
Constructs an instance of SqOverlayComponent.
Parameters :
|
backdrop | |
Type : string
|
|
Default value : 'static'
|
|
Specifies the behavior of the backdrop when clicked. |
bodyColor | |
Type : string
|
|
Default value : 'var(--background_secondary)'
|
|
The background color of the overlay body. |
borderless | |
Type : boolean
|
|
Default value : false
|
|
Determines whether the overlay has a border. |
footerColor | |
Type : string
|
|
Default value : 'var(--background_secondary)'
|
|
The background color of the overlay footer. |
headerColor | |
Type : string
|
|
Default value : 'var(--background_secondary)'
|
|
The background color of the overlay header. |
headerItemsColor | |
Type : string
|
|
Default value : ''
|
|
The text color of items within the overlay header. |
id | |
Type : string
|
|
Default value : `overlay-random-id-${(1 + Date.now() + Math.random()).toString().replace('.', '')}`
|
|
A unique identifier for the overlay. |
open | |
Type : boolean
|
|
Indicates whether the overlay is open or closed. |
overlayDirection | |
Type : "right" | "left"
|
|
Default value : 'right'
|
|
The direction in which the overlay slides in when opened. |
showClose | |
Type : boolean
|
|
Default value : true
|
|
Determines whether the close button is shown. |
width | |
Type : string
|
|
Default value : '475px'
|
|
The width of the overlay. |
leftPress | |
Type : EventEmitter<void>
|
|
Emits an event when the left arrow key is pressed. |
overlayClose | |
Type : EventEmitter<void>
|
|
Emits an event when the overlay is closed. |
rightPress | |
Type : EventEmitter<void>
|
|
Emits an event when the right arrow key is pressed. |
doCssWidth |
doCssWidth()
|
Applies CSS styles to set the width of the overlay.
Returns :
void
|
events | ||||||||
events(key: string)
|
||||||||
Handles specific keyboard events.
Parameters :
Returns :
void
|
Async ngOnChanges | ||||||||
ngOnChanges(changes: SimpleChanges)
|
||||||||
Lifecycle hook that detects changes to the 'open' input property and handles modal behavior accordingly.
Parameters :
Returns :
any
|
ngOnDestroy |
ngOnDestroy()
|
Performs actions before the component is destroyed.
Returns :
void
|
observeRouter |
observeRouter()
|
Function that init the routerObservable.
Returns :
void
|
onKeydown | ||||||||
onKeydown(event: KeyboardEvent)
|
||||||||
Handles keyboard events for the modal component.
Parameters :
Returns :
void
|
removeOverlayFromBody |
removeOverlayFromBody()
|
Removes the overlay element from document body.
Returns :
void
|
toCloseOverlay |
toCloseOverlay()
|
Closes the overlay logic.
Returns :
void
|
undoCssWidth |
undoCssWidth()
|
Removes the CSS styles that set the width of the overlay.
Returns :
void
|
document |
Type : Document
|
A reference to the Document object. |
Public documentImported |
Type : Document
|
Decorators :
@Inject(DOCUMENT)
|
- The injected Document object for DOM manipulation.
|
finishOpening |
Default value : false
|
Indicates whether the overlay has finished opening. |
Optional footerTemplate |
Type : TemplateRef<ElementRef> | null
|
Default value : null
|
Decorators :
@ContentChild('footerTemplate')
|
A reference to the footer template. |
Public getWindow |
Type : GetWindow
|
- The GetWindow service for safely accessing the window object.
|
hasFooter |
Default value : false
|
Indicates whether the overlay has a footer. |
hasHeader |
Default value : false
|
Indicates whether the overlay has a header. |
Optional headerTemplate |
Type : TemplateRef<ElementRef> | null
|
Default value : null
|
Decorators :
@ContentChild('headerTemplate')
|
A reference to the header template. |
localized |
Type : URL
|
Indicates the origin path from overlay. |
modalNumber |
Type : number
|
Default value : 0
|
The number of modal elements. |
modals |
Type : HTMLCollectionOf<Element> | undefined
|
A collection of modal elements. |
overlay |
Type : ElementRef | null
|
Default value : null
|
Decorators :
@ViewChild('overlay')
|
A reference to the overlay element. |
Public router |
Type : Router
|
- The Router service for programmatic navigation.
|
routerObservable |
Type : Subscription
|
A subscription to the router change url. |
scrollY |
Default value : this.getWindow?.window()?.scrollY
|
Indicates the scroll position of the window. |
styleId |
Default value : `overlay-style-random-id-${new Date().getTime()}-${Math.random().toString(36).substring(7)}`
|
A unique style identifier. |
import { DOCUMENT } from '@angular/common'
import {
Component,
ContentChild,
ElementRef,
EventEmitter,
Inject,
Input,
OnChanges,
OnDestroy,
Output,
SimpleChanges,
TemplateRef,
ViewChild,
} from '@angular/core'
import { sleep } from '../../helpers/sleep.helper'
import { NavigationStart, Router } from '@angular/router'
import { Subscription } from 'rxjs'
import { GetWindow } from '../../helpers/window.helper'
/**
* Represents an overlay component, an abstraction with differente style but still a modal.
*
* @example
* <sq-overlay [open]="isOverlayOpen" overlayDirection="right" (overlayClose)="onOverlayClose()">
* <ng-template #headerTemplate>
* <h2>Overlay Header</h2>
* </ng-template>
* <ng-template #footerTemplate>
* Footer
* </ng-template>
* <div>
* <!-- Your content here -->
* </div>
* </sq-overlay>
* <button (click)='isOverlayOpen = true'>Open Modal</button>
*
* @implements {OnChanges}
* @implements {OnDestroy}
*/
@Component({
selector: 'sq-overlay',
templateUrl: './sq-overlay.component.html',
styleUrls: ['./sq-overlay.component.scss'],
})
export class SqOverlayComponent implements OnChanges, OnDestroy {
/**
* A unique identifier for the overlay.
*/
@Input() id = `overlay-random-id-${(1 + Date.now() + Math.random()).toString().replace('.', '')}`
/**
* Indicates whether the overlay is open or closed.
*
*/
@Input() open?: boolean
/**
* The direction in which the overlay slides in when opened.
*
*/
@Input() overlayDirection: 'right' | 'left' = 'right'
/**
* The width of the overlay.
*
*/
@Input() width = '475px'
/**
* Determines whether the overlay has a border.
*
*/
@Input() borderless = false
/**
* The background color of the overlay header.
*
*/
@Input() headerColor = 'var(--background_secondary)'
/**
* The text color of items within the overlay header.
*
*/
@Input() headerItemsColor = ''
/**
* The background color of the overlay footer.
*
*/
@Input() footerColor = 'var(--background_secondary)'
/**
* The background color of the overlay body.
*
*/
@Input() bodyColor = 'var(--background_secondary)'
/**
* Determines whether the close button is shown.
*
*/
@Input() showClose = true
/**
* Specifies the behavior of the backdrop when clicked.
*
*/
@Input() backdrop = 'static'
/**
* Emits an event when the overlay is closed.
*
*/
@Output() overlayClose: EventEmitter<void> = new EventEmitter()
/**
* Emits an event when the left arrow key is pressed.
*
*/
@Output() leftPress: EventEmitter<void> = new EventEmitter()
/**
* Emits an event when the right arrow key is pressed.
*
*/
@Output() rightPress: EventEmitter<void> = new EventEmitter()
/**
* A reference to the overlay element.
*
*/
@ViewChild('overlay') overlay: ElementRef | null = null
/**
* A reference to the header template.
*
*/
@ContentChild('headerTemplate') headerTemplate?: TemplateRef<ElementRef> | null = null
/**
* A reference to the footer template.
*
*/
@ContentChild('footerTemplate') footerTemplate?: TemplateRef<ElementRef> | null = null
/**
* A collection of modal elements.
*
*/
modals: HTMLCollectionOf<Element> | undefined
/**
* The number of modal elements.
*
*/
modalNumber = 0
/**
* Indicates whether the overlay has a header.
*
*/
hasHeader = false
/**
* Indicates whether the overlay has a footer.
*
*/
hasFooter = false
/**
* A reference to the Document object.
*
*/
document: Document
/**
* A unique style identifier.
*
*/
styleId = `overlay-style-random-id-${new Date().getTime()}-${Math.random().toString(36).substring(7)}`
/**
* Indicates whether the overlay has finished opening.
*
*/
finishOpening = false
/**
* Indicates the origin path from overlay.
*
*/
localized: URL
/**
* A subscription to the router change url.
*/
routerObservable!: Subscription
/**
* Indicates the scroll position of the window.
*/
scrollY = this.getWindow?.window()?.scrollY
/**
* Constructs an instance of SqOverlayComponent.
* @constructor
* @param {Document} documentImported - The injected Document object for DOM manipulation.
* @param {Router} router - The Router service for programmatic navigation.
* @param {GetWindow} getWindow - The GetWindow service for safely accessing the window object.
*/
constructor(@Inject(DOCUMENT) public documentImported: Document, public router: Router, public getWindow: GetWindow) {
this.onKeydown = this.onKeydown.bind(this)
this.document = documentImported || document
this.localized = new URL(this.getWindow.href())
}
/**
* Lifecycle hook that detects changes to the 'open' input property and handles modal behavior accordingly.
*
* @param changes - The changes detected in the component's input properties.
*/
async ngOnChanges(changes: SimpleChanges) {
if (changes.hasOwnProperty('width') && this.open) {
this.doCssWidth()
}
if (changes.hasOwnProperty('open')) {
const overlay = this.overlay
if (overlay) {
const body = this.document.getElementsByTagName('body')[0]
const backdrop = this.document.getElementById('modal-backdrop') || this.document.createElement('div')
if (this.open) {
this.scrollY = this.getWindow?.window()?.scrollY
body.appendChild(overlay.nativeElement)
this.observeRouter()
this.doCssWidth()
this.hasFooter = !!this.footerTemplate
this.hasHeader = !!this.headerTemplate
body.classList.add('block')
overlay.nativeElement.style.display = 'flex'
this.getWindow?.window()?.addEventListener('keydown', this.onKeydown)
this.modals = this.document.getElementsByClassName('modal open')
await sleep(10)
this.modalNumber = this.modals?.length || 0
if (this.modalNumber <= 1) {
backdrop.setAttribute('id', 'modal-backdrop')
backdrop.setAttribute('class', 'modal-backdrop show')
body.appendChild(backdrop)
} else if (this.modalNumber > 1) {
overlay.nativeElement.style.zIndex = 1060 + this.modalNumber + 1
backdrop.setAttribute('style', `z-index: ${1060 + this.modalNumber};`)
}
this.finishOpening = true
} else {
this.removeOverlayFromBody()
}
}
}
}
/**
* Performs actions before the component is destroyed.
*/
ngOnDestroy(): void {
this.routerObservable?.unsubscribe()
}
/**
* Function that init the routerObservable.
*/
observeRouter() {
this.routerObservable = this.router.events.subscribe(async (event) => {
if (event instanceof NavigationStart) {
const destinationRoute = new URL(event.url, this.localized.origin)
if ((this.localized.origin + this.localized.pathname) !== (destinationRoute.origin + destinationRoute.pathname)) {
this.removeOverlayFromBody()
await sleep(1000)
}
}
})
}
/**
* Removes the overlay element from document body.
*/
removeOverlayFromBody() {
const body = this.document.getElementsByTagName('body')[0]
if (this.modalNumber <= 1) {
body?.classList?.remove('block')
if (window.scrollY !== this.scrollY) {
if (this.scrollY) this.getWindow?.window()?.scrollTo(0, this.scrollY)
}
}
const backdrop = this.document.getElementById('modal-backdrop')
const overlay: any = this.document.getElementById(this.id)
this.overlayClose.emit()
this.finishOpening = false
this.undoCssWidth()
overlay?.parentNode?.removeChild(overlay)
if (this.modalNumber === 2) {
backdrop?.removeAttribute('style')
} else if (this.modalNumber <= 1) {
backdrop?.parentNode?.removeChild(backdrop)
}
window.removeEventListener('keydown', this.onKeydown)
}
/**
* Applies CSS styles to set the width of the overlay.
*/
doCssWidth() {
const css = `
.overlay.open .modal-dialog.opened {
width: ${this.width};
}
`
const head = this.document.getElementsByTagName('head')[0]
let style = this.document.getElementById(this.styleId)
if (!style) {
style = this.document.createElement('style')
style.setAttribute('id', this.styleId)
style.appendChild(this.document.createTextNode(css))
head.appendChild(style)
} else {
style.innerHTML = ''
style.appendChild(this.document.createTextNode(css))
}
}
/**
* Removes the CSS styles that set the width of the overlay.
*/
undoCssWidth() {
const style = this.document.getElementById(this.styleId)
if (style?.parentNode) {
style.parentNode.removeChild(style)
}
}
/**
* Handles keyboard events for the modal component.
*
* @param event - The keyboard event object.
*/
onKeydown(event: KeyboardEvent) {
if (this.open) {
this.modals = this.document.getElementsByClassName('modal open')
if (this.modals?.length && this.modals[this.modals.length - 1]?.id === this.id) {
this.events(event.key)
}
}
}
/**
* Handles specific keyboard events.
*
* @param key - The key code of the pressed key.
*/
events(key: string) {
switch (key) {
case 'Escape':
this.overlayClose.emit()
break
case 'ArrowLeft':
this.leftPress.emit()
break
case 'ArrowRight':
this.rightPress.emit()
break
}
}
/**
* Closes the overlay logic.
*/
toCloseOverlay() {
if (this.overlay && this.open) {
const body = this.document.getElementsByTagName('body')[0]
const backdrop = this.document.getElementById('modal-backdrop') || this.document.createElement('div')
this.overlayClose.emit()
this.overlay.nativeElement.style.display = 'none'
if (backdrop.parentNode && this.modalNumber <= 1) {
backdrop.parentNode.removeChild(backdrop)
body.classList.remove('block')
}
window.removeEventListener('keydown', this.onKeydown)
}
}
}
<div [id]="id" class="modal overlay {{ overlayDirection }}" [ngClass]="{ open: open }" #overlay>
<div
class="modal-dialog {{ overlayDirection }}"
[ngStyle]="{ 'background-color': headerColor }"
[ngClass]="{ opened: finishOpening }"
[clickOutsideEnabled]="!!(open && backdrop !== 'static' && overlay)"
(clickOutside)="toCloseOverlay()"
>
<div class="modal-content scrollbar">
<div
class="modal-header"
[ngClass]="{
'without-header': !hasHeader,
borderless: borderless
}"
[ngStyle]="{
'background-color': headerColor
}"
>
<ng-container *ngIf="headerTemplate">
<ng-container *ngTemplateOutlet="headerTemplate"></ng-container>
</ng-container>
<button *ngIf="showClose" id="close-button" type="button" class="close button-close" aria-label="Close" (click)="overlayClose.emit()">
<i class="fa-solid fa-xmark" [ngStyle]="{ color: headerItemsColor }"></i>
</button>
</div>
<div
class="modal-body scrollbar"
[ngClass]="{
'without-footer': !hasFooter
}"
[ngStyle]="{
'background-color': bodyColor
}"
>
<ng-content></ng-content>
</div>
<div
class="modal-footer"
*ngIf="hasFooter"
[ngClass]="{
borderless: borderless
}"
[ngStyle]="{
'background-color': footerColor
}"
>
<ng-container *ngIf="footerTemplate">
<ng-container *ngTemplateOutlet="footerTemplate"></ng-container>
</ng-container>
</div>
</div>
</div>
</div>
./sq-overlay.component.scss
::ng-deep {
.overlay {
padding: 0;
border-radius: 0px;
&.left {
justify-content: flex-start;
}
&.right {
justify-content: flex-end;
}
.modal-dialog {
width: 300px;
max-width: 100vw;
height: 100%;
margin: 0;
position: fixed;
transition: opacity 0.3s linear, left 0.3s ease-out, right 0.3s ease-out,
width 0.3s ease-out, height 0.3s ease-out !important;
transform: translate3d(0%, 0, 0) !important;
.modal-content {
height: 100%;
border-radius: 0;
overflow-y: auto;
border: none;
background-color: transparent;
.modal-header {
display: flex;
justify-content: space-between;
align-items: center;
height: 60px;
border-radius: 0;
overflow: hidden;
padding: 1rem 2rem;
&.without-header {
padding: 0 0.08rem;
border: none;
justify-content: flex-end;
.button-close {
margin: 0;
padding: 0;
}
}
.button-close {
max-height: 30px;
line-height: 30px;
opacity: 0.7 !important;
transition: var(--transition);
z-index: 1;
span {
font-size: 2.1rem;
font-weight: 300;
transition: var(--transition);
}
&:hover,
&:focus {
outline: none;
span {
font-weight: 500;
}
}
}
}
.modal-body {
height: calc(100vh - 112px);
padding: 2rem;
overflow-y: auto;
overflow-x: hidden;
&.without-footer {
height: calc(100vh - 52px);
}
}
.modal-footer {
height: 60px;
border-radius: 0;
overflow: hidden;
padding: 0.5rem 2rem;
}
.modal-footer > * {
margin: 0;
}
.borderless {
border-color: transparent;
}
}
}
}
}