Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ESLPopup: introduce config for easily acessing params #2823

Open
wants to merge 9 commits into
base: main-beta
Choose a base branch
from
3 changes: 0 additions & 3 deletions src/modules/esl-alert/core/esl-alert.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,9 +18,6 @@ export interface ESLAlertActionParams extends ESLToggleableRequestDetails {
hideTime?: number;
}

/** @deprecated alias, use {@link ESLAlertActionParams} instead. Will be removed in v5.0.0. */
export type AlertActionParams = ESLAlertActionParams;

/**
* ESLAlert component
* @author Julia Murashko, Alexey Stsefanovich (ala'n)
Expand Down
3 changes: 0 additions & 3 deletions src/modules/esl-panel/core/esl-panel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,9 +15,6 @@ export interface ESLPanelActionParams extends ESLToggleableActionParams {
noAnimate?: boolean;
}

/** @deprecated alias, use {@link ESLPanelActionParams} instead. Will be removed in v5.0.0. */
export type PanelActionParams = ESLPanelActionParams;

/**
* ESLPanel component
* @author Julia Murashko
Expand Down
1 change: 1 addition & 0 deletions src/modules/esl-popup/core.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,3 +2,4 @@ export type {ESLPopupTagShape} from './core/esl-popup.shape';

export * from './core/esl-popup';
export * from './core/esl-popup-placeholder';
export * from './core/esl-popup-types';
54 changes: 54 additions & 0 deletions src/modules/esl-popup/core/esl-popup-types.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
import type {ESLToggleableActionParams} from '../../esl-toggleable/core';
import type {PositionType, PositionOriginType} from './esl-popup-position';

export interface ESLPopupActionParams extends ESLToggleableActionParams {
/** popup position relative to trigger */
position?: PositionType;
/** clarification of the popup position, whether it should start on the outside of trigger or the inside of trigger */
positionOrigin?: PositionOriginType;
/** popup behavior if it does not fit in the window */
behavior?: string;
/** Disable hiding the popup depending on the visibility of the activator */
disableActivatorObservation?: boolean;
/** Margins on the edges of the arrow. */
marginArrow?: number;
/** Offset of the arrow as a percentage of the popup edge (0% - at the left edge, 100% - at the right edge, for RTL it is vice versa) */
offsetArrow?: string;
/** Offset in pixels from the trigger element */
offsetTrigger?: number;
/**
* Offset in pixels from the edges of the container (or window if the container is not defined)
* value as a number for equals x and y offsets
* value as an array for different x and y offsets
*/
offsetContainer?: number | [number, number];
/** Margin around the element that is used as the viewport for checking the visibility of the popup activator */
intersectionMargin?: string;
/** Target to container element to define bounds of popups visibility */
container?: string;
/** Container element that defines bounds of popups visibility (is not taken into account if the container attr is set on popup) */
containerEl?: HTMLElement;

/** Extra class to add to popup on activation */
extraClass?: string;
/** Extra styles to add to popup on activation */
extraStyle?: string;
}

export type ProxiedParams = Required<ESLPopupActionParams>;

/** List of ESLPopupActionParams keys */
export const KEYSOF_POPUP_ACTION_PARAMS: (keyof ESLPopupActionParams)[] = [
'position',
'positionOrigin',
'behavior',
'disableActivatorObservation',
'marginArrow',
'offsetArrow',
'offsetTrigger',
'offsetContainer',
'intersectionMargin',
'container',
'containerEl',
'extraClass',
'extraStyle'] as const;
138 changes: 54 additions & 84 deletions src/modules/esl-popup/core/esl-popup.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,54 +6,18 @@ import {afterNextRender, rafDecorator} from '../../esl-utils/async/raf';
import {ESLToggleable} from '../../esl-toggleable/core';
import {isElement, isRelativeNode, isRTL, Rect, getListScrollParents, getViewportRect} from '../../esl-utils/dom';
import {parseBoolean, parseNumber, toBooleanAttribute} from '../../esl-utils/misc/format';
import {copyDefinedKeys} from '../../esl-utils/misc/object';
import {ESLIntersectionTarget, ESLIntersectionEvent} from '../../esl-event-listener/core/targets/intersection.target';
import {calcPopupPosition, isOnHorizontalAxis} from './esl-popup-position';
import {ESLPopupPlaceholder} from './esl-popup-placeholder';
import {KEYSOF_POPUP_ACTION_PARAMS} from './esl-popup-types';

import type {ESLToggleableActionParams, ESLA11yType} from '../../esl-toggleable/core';
import type {PopupPositionConfig, PositionType, PositionOriginType, IntersectionRatioRect} from './esl-popup-position';
import type {ESLPopupActionParams, ProxiedParams} from './esl-popup-types';

const INTERSECTION_LIMIT_FOR_ADJACENT_AXIS = 0.7;
const DEFAULT_OFFSET_ARROW = 50;

export interface ESLPopupActionParams extends ESLToggleableActionParams {
/** popup position relative to trigger */
position?: PositionType;
/** clarification of the popup position, whether it should start on the outside of trigger or the inside of trigger */
positionOrigin?: PositionOriginType;
/** popup behavior if it does not fit in the window */
behavior?: string;
/** Disable hiding the popup depending on the visibility of the activator */
disableActivatorObservation?: boolean;
/** Margins on the edges of the arrow. */
marginArrow?: number;
/** Offset of the arrow as a percentage of the popup edge (0% - at the left edge, 100% - at the right edge, for RTL it is vice versa) */
offsetArrow?: string;
/** Offset in pixels from the trigger element */
offsetTrigger?: number;
/**
* Offset in pixels from the edges of the container (or window if the container is not defined)
* value as a number for equals x and y offsets
* value as an array for different x and y offsets
*/
offsetContainer?: number | [number, number];
/** Margin around the element that is used as the viewport for checking the visibility of the popup activator */
intersectionMargin?: string;
/** Target to container element to define bounds of popups visibility */
container?: string;
/** Container element that defines bounds of popups visibility (is not taken into account if the container attr is set on popup) */
containerEl?: HTMLElement;

/** Extra class to add to popup on activation */
extraClass?: string;
/** Extra styles to add to popup on activation */
extraStyle?: string;
}

/** @deprecated alias, use {@link ESLPopupActionParams} instead, will be removed in v5.0.0 */
export type PopupActionParams = ESLPopupActionParams;

@ExportNs('Popup')
export class ESLPopup extends ESLToggleable {
public static override is = 'esl-popup';
Expand All @@ -64,6 +28,9 @@ export class ESLPopup extends ESLToggleable {
intersectionMargin: '0px'
};

/** List of action params keys */
public static PARAM_KEYS: string[] = KEYSOF_POPUP_ACTION_PARAMS as string[];
dshovchko marked this conversation as resolved.
Show resolved Hide resolved

/** Classname of popups arrow element */
@attr({defaultValue: 'esl-popup-arrow'}) public arrowClass: string;

Expand Down Expand Up @@ -116,12 +83,30 @@ export class ESLPopup extends ESLToggleable {

public $placeholder: ESLPopupPlaceholder | null;

protected _extraClass?: string;
protected _extraStyle?: string;
@memoize()
public get config(): ProxiedParams {
// eslint-disable-next-line @typescript-eslint/no-this-alias
const $popup = this;
return new Proxy<ProxiedParams>({} as ProxiedParams, {
get<T extends keyof ESLPopupActionParams>(target: ESLPopupActionParams, p: T | symbol): ESLPopupActionParams[T] {
return Reflect.get($popup._params, p) ?? Reflect.get($popup, p);
},
set: (): boolean => false,
deleteProperty: (): boolean => false,
has<T extends keyof ESLPopupActionParams>(target: ESLPopupActionParams, p: T | symbol): boolean {
return Reflect.has($popup._params, p) || Reflect.has($popup, p);
},
ownKeys(target: ESLPopupActionParams): (string | symbol)[] {
const paramKeys = Reflect.ownKeys($popup._params);
const popupKeys = ($popup.constructor as typeof ESLPopup).PARAM_KEYS
.filter((key: string) => !paramKeys.includes(key) && Reflect.has($popup, key as keyof ESLPopup));
return [...paramKeys, ...popupKeys];
},
getOwnPropertyDescriptor: (): PropertyDescriptor | undefined => ({enumerable: true, configurable: true})
});
}

protected _containerEl?: HTMLElement;
protected _offsetContainer: number | [number, number];
protected _intersectionMargin: string;
protected _params: ESLPopupActionParams = {};
protected _intersectionRatio: IntersectionRatioRect = {};
protected _updateLoopID: number;

Expand All @@ -134,7 +119,8 @@ export class ESLPopup extends ESLToggleable {
/** Container element that define bounds of popups visibility */
@memoize()
protected get $container(): HTMLElement | undefined {
return this.container ? ESLTraversingQuery.first(this.container, this) as HTMLElement : this._containerEl;
const {container} = this.config;
return container ? ESLTraversingQuery.first(container, this) as HTMLElement : this.config.containerEl;
}

/** Get the size and position of the container */
Expand All @@ -157,7 +143,7 @@ export class ESLPopup extends ESLToggleable {
/** Get offsets arrow ratio */
@memoize()
protected get offsetArrowRatio(): number {
const offset = parseNumber(this.offsetArrow, DEFAULT_OFFSET_ARROW);
const offset = parseNumber(this.config.offsetArrow, DEFAULT_OFFSET_ARROW);
const offsetNormalized = Math.max(0, Math.min(offset, 100));
const ratio = offsetNormalized / 100;
return isRTL(this) ? 1 - ratio : ratio;
Expand Down Expand Up @@ -205,24 +191,7 @@ export class ESLPopup extends ESLToggleable {
}

super.onShow(params);

// TODO: change flow to use merged params unless attribute state is used in CSS
Object.assign(this, copyDefinedKeys({
position: params.position,
positionOrigin: params.positionOrigin,
behavior: params.behavior,
container: params.container,
marginArrow: params.marginArrow,
offsetArrow: params.offsetArrow,
offsetTrigger: params.offsetTrigger,
disableActivatorObservation: params.disableActivatorObservation
}));

this._extraClass = params.extraClass;
this._extraStyle = params.extraStyle;
this._containerEl = params.containerEl;
this._offsetContainer = params.offsetContainer || 0;
this._intersectionMargin = params.intersectionMargin || '0px';
this._params = params;

this.style.visibility = 'hidden'; // eliminates the blinking of the popup at the previous position

Expand All @@ -249,8 +218,8 @@ export class ESLPopup extends ESLToggleable {
this._updatePosition();

this.style.visibility = 'visible';
this.style.cssText += this._extraStyle || '';
this.$$cls(this._extraClass || '', true);
this.style.cssText += this.config.extraStyle || '';
this.$$cls(this.config.extraClass || '', true);

this.$$on(this._onActivatorScroll);
this.$$on(this._onActivatorIntersection);
Expand All @@ -273,7 +242,7 @@ export class ESLPopup extends ESLToggleable {
this._stopUpdateLoop();

this.$$attr('style', '');
this.$$cls(this._extraClass || '', false);
this.$$cls(this.config.extraClass || '', false);

this.$$off({group: 'observer'});

Expand All @@ -289,7 +258,7 @@ export class ESLPopup extends ESLToggleable {

protected get intersectionOptions(): IntersectionObserverInit {
return {
rootMargin: this._intersectionMargin,
rootMargin: this.config.intersectionMargin,
threshold: range(9, (x) => x / 8)
};
}
Expand All @@ -300,7 +269,7 @@ export class ESLPopup extends ESLToggleable {
group: 'observer',
event: ESLIntersectionEvent.TYPE,
target: ($popup: ESLPopup) => $popup.activator ? ESLIntersectionTarget.for($popup.activator, $popup.intersectionOptions) : [],
condition: ($popup: ESLPopup) => !$popup.disableActivatorObservation
condition: ($popup: ESLPopup) => !$popup.config.disableActivatorObservation
})
protected _onActivatorIntersection(event: ESLIntersectionEvent): void {
this._intersectionRatio = {};
Expand All @@ -309,7 +278,7 @@ export class ESLPopup extends ESLToggleable {
return;
}

const isHorizontal = isOnHorizontalAxis(this.position);
const isHorizontal = isOnHorizontalAxis(this.config.position);
const checkIntersection = (isMajorAxis: boolean, intersectionRatio: number): void => {
if (isMajorAxis && intersectionRatio < INTERSECTION_LIMIT_FOR_ADJACENT_AXIS) this.hide();
};
Expand Down Expand Up @@ -397,27 +366,28 @@ export class ESLPopup extends ESLToggleable {
}

protected get positionConfig(): PopupPositionConfig {
const popupRect = Rect.from(this);
const arrowRect = this.$arrow ? Rect.from(this.$arrow) : new Rect();
const triggerRect = this.activator ? Rect.from(this.activator).shift(window.scrollX, window.scrollY) : new Rect();
const {containerRect} = this;
const {$arrow, activator, containerRect, offsetArrowRatio} = this;
const {position, positionOrigin, behavior, marginArrow, offsetContainer, offsetTrigger} = this.config;

const innerMargin = this.offsetTrigger + arrowRect.width / 2;
const popupRect = Rect.from(this);
const arrowRect = $arrow ? Rect.from($arrow) : new Rect();
const triggerRect = activator ? Rect.from(activator).shift(window.scrollX, window.scrollY) : new Rect();

const innerMargin = offsetTrigger + arrowRect.width / 2;
return {
position: this.position,
hasInnerOrigin: this.positionOrigin === 'inner',
behavior: this.behavior,
marginArrow: this.marginArrow,
offsetArrowRatio: this.offsetArrowRatio,
position,
hasInnerOrigin: positionOrigin === 'inner',
behavior,
marginArrow,
offsetArrowRatio,
intersectionRatio: this._intersectionRatio,
arrow: arrowRect,
element: popupRect,
trigger: triggerRect,
inner: this.positionOrigin === 'inner' ? triggerRect.shrink(innerMargin) : triggerRect.grow(innerMargin),
outer: (typeof this._offsetContainer === 'number') ?
containerRect.shrink(this._offsetContainer) :
containerRect.shrink(...this._offsetContainer),
inner: positionOrigin === 'inner' ? triggerRect.shrink(innerMargin) : triggerRect.grow(innerMargin),
outer: (typeof offsetContainer === 'number') ?
containerRect.shrink(offsetContainer) :
containerRect.shrink(...offsetContainer),
isRTL: isRTL(this)
};
}
Expand All @@ -434,7 +404,7 @@ export class ESLPopup extends ESLToggleable {
this.style.top = `${popup.y}px`;
if (!this.$arrow) return;
// set arrow position
const isHorizontal = isOnHorizontalAxis(this.position);
const isHorizontal = isOnHorizontalAxis(this.config.position);
this.$arrow.style.left = isHorizontal ? '' : `${arrow.x}px`;
this.$arrow.style.top = isHorizontal ? `${arrow.y}px` : '';
}
Expand Down
Loading
Loading