@ng-bootstrap/ng-bootstrap
Version:
Angular powered Bootstrap
182 lines • 23.5 kB
JavaScript
import { arrow, createPopperLite, flip, preventOverflow, } from '@popperjs/core';
import { NgbRTL } from './rtl';
import { inject } from '@angular/core';
const placementSeparator = /\s+/;
const spacesRegExp = / +/gi;
/**
* Matching classes from the Bootstrap ones to the poppers ones.
* The first index of each array is used for the left to right direction,
* the second one is used for the right to left, defaulting to the first index (when LTR and RTL lead to the same class)
*
* See [Bootstrap alignments](https://getbootstrap.com/docs/5.1/components/dropdowns/#alignment-options)
* and [Popper placements](https://popper.js.org/docs/v2/constructors/#options)
*/
const bootstrapPopperMatches = {
top: ['top'],
bottom: ['bottom'],
start: ['left', 'right'],
left: ['left'],
end: ['right', 'left'],
right: ['right'],
'top-start': ['top-start', 'top-end'],
'top-left': ['top-start'],
'top-end': ['top-end', 'top-start'],
'top-right': ['top-end'],
'bottom-start': ['bottom-start', 'bottom-end'],
'bottom-left': ['bottom-start'],
'bottom-end': ['bottom-end', 'bottom-start'],
'bottom-right': ['bottom-end'],
'start-top': ['left-start', 'right-start'],
'left-top': ['left-start'],
'start-bottom': ['left-end', 'right-end'],
'left-bottom': ['left-end'],
'end-top': ['right-start', 'left-start'],
'right-top': ['right-start'],
'end-bottom': ['right-end', 'left-end'],
'right-bottom': ['right-end'],
};
export function getPopperClassPlacement(placement, isRTL) {
const [leftClass, rightClass] = bootstrapPopperMatches[placement];
return isRTL ? rightClass || leftClass : leftClass;
}
const popperStartPrimaryPlacement = /^left/;
const popperEndPrimaryPlacement = /^right/;
const popperStartSecondaryPlacement = /^start/;
const popperEndSecondaryPlacement = /^end/;
export function getBootstrapBaseClassPlacement(baseClass, placement) {
let [primary, secondary] = placement.split('-');
const newPrimary = primary.replace(popperStartPrimaryPlacement, 'start').replace(popperEndPrimaryPlacement, 'end');
let classnames = [newPrimary];
if (secondary) {
let newSecondary = secondary;
if (primary === 'left' || primary === 'right') {
newSecondary = newSecondary
.replace(popperStartSecondaryPlacement, 'top')
.replace(popperEndSecondaryPlacement, 'bottom');
}
classnames.push(`${newPrimary}-${newSecondary}`);
}
if (baseClass) {
classnames = classnames.map((classname) => `${baseClass}-${classname}`);
}
return classnames.join(' ');
}
/*
* Accept the placement array and applies the appropriate placement dependent on the viewport.
* Returns the applied placement.
* In case of auto placement, placements are selected in order
* 'top', 'bottom', 'start', 'end',
* 'top-start', 'top-end',
* 'bottom-start', 'bottom-end',
* 'start-top', 'start-bottom',
* 'end-top', 'end-bottom'.
* */
export function getPopperOptions({ placement, baseClass }, rtl) {
let placementVals = Array.isArray(placement)
? placement
: placement.split(placementSeparator);
// No need to consider left and right here, as start and end are enough, and it is used for 'auto' placement only
const allowedPlacements = [
'top',
'bottom',
'start',
'end',
'top-start',
'top-end',
'bottom-start',
'bottom-end',
'start-top',
'start-bottom',
'end-top',
'end-bottom',
];
// replace auto placement with other placements
let hasAuto = placementVals.findIndex((val) => val === 'auto');
if (hasAuto >= 0) {
allowedPlacements.forEach(function (obj) {
if (placementVals.find((val) => val.search('^' + obj) !== -1) == null) {
placementVals.splice(hasAuto++, 1, obj);
}
});
}
const popperPlacements = placementVals.map((_placement) => {
return getPopperClassPlacement(_placement, rtl.isRTL());
});
let mainPlacement = popperPlacements.shift();
const bsModifier = {
name: 'bootstrapClasses',
enabled: !!baseClass,
phase: 'write',
fn({ state }) {
const bsClassRegExp = new RegExp(baseClass + '(-[a-z]+)*', 'gi');
const popperElement = state.elements.popper;
const popperPlacement = state.placement;
let className = popperElement.className;
// Remove old bootstrap classes
className = className.replace(bsClassRegExp, '');
// Add current placements
className += ` ${getBootstrapBaseClassPlacement(baseClass, popperPlacement)}`;
// Remove multiple spaces
className = className.trim().replace(spacesRegExp, ' ');
// Reassign
popperElement.className = className;
},
};
return {
placement: mainPlacement,
modifiers: [
bsModifier,
flip,
preventOverflow,
arrow,
{
enabled: true,
name: 'flip',
options: {
fallbackPlacements: popperPlacements,
},
},
{
enabled: true,
name: 'preventOverflow',
phase: 'main',
fn: function () { },
},
],
};
}
function noop(arg) {
return arg;
}
export function ngbPositioning() {
const rtl = inject(NgbRTL);
let popperInstance = null;
return {
createPopper(positioningOption) {
if (!popperInstance) {
const updatePopperOptions = positioningOption.updatePopperOptions || noop;
let popperOptions = updatePopperOptions(getPopperOptions(positioningOption, rtl));
popperInstance = createPopperLite(positioningOption.hostElement, positioningOption.targetElement, popperOptions);
}
},
update() {
if (popperInstance) {
popperInstance.update();
}
},
setOptions(positioningOption) {
if (popperInstance) {
const updatePopperOptions = positioningOption.updatePopperOptions || noop;
let popperOptions = updatePopperOptions(getPopperOptions(positioningOption, rtl));
popperInstance.setOptions(popperOptions);
}
},
destroy() {
if (popperInstance) {
popperInstance.destroy();
popperInstance = null;
}
},
};
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"positioning.js","sourceRoot":"","sources":["../../../../src/util/positioning.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tarrow,\n\tcreatePopperLite,\n\tflip,\n\tInstance,\n\tModifier,\n\tPlacement as PopperPlacement,\n\tpreventOverflow,\n\tOptions,\n} from '@popperjs/core';\nimport { NgbRTL } from './rtl';\nimport { inject } from '@angular/core';\n\nconst placementSeparator = /\\s+/;\nconst spacesRegExp = /  +/gi;\n\n/**\n * Matching classes from the Bootstrap ones to the poppers ones.\n * The first index of each array is used for the left to right direction,\n * the second one is used for the right to left, defaulting to the first index (when LTR and RTL lead to the same class)\n *\n * See [Bootstrap alignments](https://getbootstrap.com/docs/5.1/components/dropdowns/#alignment-options)\n * and [Popper placements](https://popper.js.org/docs/v2/constructors/#options)\n */\nconst bootstrapPopperMatches = {\n\ttop: ['top'],\n\tbottom: ['bottom'],\n\tstart: ['left', 'right'],\n\tleft: ['left'],\n\tend: ['right', 'left'],\n\tright: ['right'],\n\t'top-start': ['top-start', 'top-end'],\n\t'top-left': ['top-start'],\n\t'top-end': ['top-end', 'top-start'],\n\t'top-right': ['top-end'],\n\t'bottom-start': ['bottom-start', 'bottom-end'],\n\t'bottom-left': ['bottom-start'],\n\t'bottom-end': ['bottom-end', 'bottom-start'],\n\t'bottom-right': ['bottom-end'],\n\t'start-top': ['left-start', 'right-start'],\n\t'left-top': ['left-start'],\n\t'start-bottom': ['left-end', 'right-end'],\n\t'left-bottom': ['left-end'],\n\t'end-top': ['right-start', 'left-start'],\n\t'right-top': ['right-start'],\n\t'end-bottom': ['right-end', 'left-end'],\n\t'right-bottom': ['right-end'],\n};\n\nexport function getPopperClassPlacement(placement: Placement, isRTL: boolean): PopperPlacement {\n\tconst [leftClass, rightClass] = bootstrapPopperMatches[placement];\n\treturn isRTL ? rightClass || leftClass : leftClass;\n}\n\nconst popperStartPrimaryPlacement = /^left/;\nconst popperEndPrimaryPlacement = /^right/;\nconst popperStartSecondaryPlacement = /^start/;\nconst popperEndSecondaryPlacement = /^end/;\nexport function getBootstrapBaseClassPlacement(baseClass: string, placement: PopperPlacement): string {\n\tlet [primary, secondary] = placement.split('-');\n\tconst newPrimary = primary.replace(popperStartPrimaryPlacement, 'start').replace(popperEndPrimaryPlacement, 'end');\n\tlet classnames = [newPrimary];\n\tif (secondary) {\n\t\tlet newSecondary = secondary;\n\t\tif (primary === 'left' || primary === 'right') {\n\t\t\tnewSecondary = newSecondary\n\t\t\t\t.replace(popperStartSecondaryPlacement, 'top')\n\t\t\t\t.replace(popperEndSecondaryPlacement, 'bottom');\n\t\t}\n\t\tclassnames.push(`${newPrimary}-${newSecondary}` as Placement);\n\t}\n\tif (baseClass) {\n\t\tclassnames = classnames.map((classname) => `${baseClass}-${classname}`);\n\t}\n\treturn classnames.join(' ');\n}\n\n/*\n * Accept the placement array and applies the appropriate placement dependent on the viewport.\n * Returns the applied placement.\n * In case of auto placement, placements are selected in order\n *   'top', 'bottom', 'start', 'end',\n *   'top-start', 'top-end',\n *   'bottom-start', 'bottom-end',\n *   'start-top', 'start-bottom',\n *   'end-top', 'end-bottom'.\n * */\nexport function getPopperOptions({ placement, baseClass }: PositioningOptions, rtl: NgbRTL): Partial<Options> {\n\tlet placementVals: Array<Placement> = Array.isArray(placement)\n\t\t? placement\n\t\t: (placement.split(placementSeparator) as Array<Placement>);\n\n\t// No need to consider left and right here, as start and end are enough, and it is used for 'auto' placement only\n\tconst allowedPlacements = [\n\t\t'top',\n\t\t'bottom',\n\t\t'start',\n\t\t'end',\n\t\t'top-start',\n\t\t'top-end',\n\t\t'bottom-start',\n\t\t'bottom-end',\n\t\t'start-top',\n\t\t'start-bottom',\n\t\t'end-top',\n\t\t'end-bottom',\n\t];\n\n\t// replace auto placement with other placements\n\tlet hasAuto = placementVals.findIndex((val) => val === 'auto');\n\tif (hasAuto >= 0) {\n\t\tallowedPlacements.forEach(function (obj) {\n\t\t\tif (placementVals.find((val) => val.search('^' + obj) !== -1) == null) {\n\t\t\t\tplacementVals.splice(hasAuto++, 1, obj as Placement);\n\t\t\t}\n\t\t});\n\t}\n\n\tconst popperPlacements = placementVals.map((_placement) => {\n\t\treturn getPopperClassPlacement(_placement, rtl.isRTL());\n\t});\n\n\tlet mainPlacement = popperPlacements.shift();\n\n\tconst bsModifier: Partial<Modifier<any, any>> = {\n\t\tname: 'bootstrapClasses',\n\t\tenabled: !!baseClass,\n\t\tphase: 'write',\n\t\tfn({ state }) {\n\t\t\tconst bsClassRegExp = new RegExp(baseClass + '(-[a-z]+)*', 'gi');\n\n\t\t\tconst popperElement: HTMLElement = state.elements.popper as HTMLElement;\n\t\t\tconst popperPlacement = state.placement;\n\n\t\t\tlet className = popperElement.className;\n\n\t\t\t// Remove old bootstrap classes\n\t\t\tclassName = className.replace(bsClassRegExp, '');\n\n\t\t\t// Add current placements\n\t\t\tclassName += ` ${getBootstrapBaseClassPlacement(baseClass!, popperPlacement)}`;\n\n\t\t\t// Remove multiple spaces\n\t\t\tclassName = className.trim().replace(spacesRegExp, ' ');\n\n\t\t\t// Reassign\n\t\t\tpopperElement.className = className;\n\t\t},\n\t};\n\n\treturn {\n\t\tplacement: mainPlacement,\n\t\tmodifiers: [\n\t\t\tbsModifier,\n\t\t\tflip,\n\t\t\tpreventOverflow,\n\t\t\tarrow,\n\t\t\t{\n\t\t\t\tenabled: true,\n\t\t\t\tname: 'flip',\n\t\t\t\toptions: {\n\t\t\t\t\tfallbackPlacements: popperPlacements,\n\t\t\t\t},\n\t\t\t},\n\t\t\t{\n\t\t\t\tenabled: true,\n\t\t\t\tname: 'preventOverflow',\n\t\t\t\tphase: 'main',\n\t\t\t\tfn: function () {},\n\t\t\t},\n\t\t],\n\t};\n}\n\nexport type Placement =\n\t| 'auto'\n\t| 'top'\n\t| 'bottom'\n\t| 'start'\n\t| 'left'\n\t| 'end'\n\t| 'right'\n\t| 'top-start'\n\t| 'top-left'\n\t| 'top-end'\n\t| 'top-right'\n\t| 'bottom-start'\n\t| 'bottom-left'\n\t| 'bottom-end'\n\t| 'bottom-right'\n\t| 'start-top'\n\t| 'left-top'\n\t| 'start-bottom'\n\t| 'left-bottom'\n\t| 'end-top'\n\t| 'right-top'\n\t| 'end-bottom'\n\t| 'right-bottom';\n\nexport type PlacementArray = Placement | Array<Placement> | string;\n\ninterface PositioningOptions {\n\thostElement: HTMLElement;\n\ttargetElement: HTMLElement;\n\tplacement: string | Placement | PlacementArray;\n\tappendToBody?: boolean;\n\tbaseClass?: string;\n\tupdatePopperOptions?: (options: Partial<Options>) => Partial<Options>;\n}\n\nfunction noop(arg) {\n\treturn arg;\n}\n\nexport function ngbPositioning() {\n\tconst rtl = inject(NgbRTL);\n\tlet popperInstance: Instance | null = null;\n\n\treturn {\n\t\tcreatePopper(positioningOption: PositioningOptions) {\n\t\t\tif (!popperInstance) {\n\t\t\t\tconst updatePopperOptions = positioningOption.updatePopperOptions || noop;\n\t\t\t\tlet popperOptions = updatePopperOptions(getPopperOptions(positioningOption, rtl));\n\t\t\t\tpopperInstance = createPopperLite(\n\t\t\t\t\tpositioningOption.hostElement,\n\t\t\t\t\tpositioningOption.targetElement,\n\t\t\t\t\tpopperOptions,\n\t\t\t\t);\n\t\t\t}\n\t\t},\n\t\tupdate() {\n\t\t\tif (popperInstance) {\n\t\t\t\tpopperInstance.update();\n\t\t\t}\n\t\t},\n\t\tsetOptions(positioningOption: PositioningOptions) {\n\t\t\tif (popperInstance) {\n\t\t\t\tconst updatePopperOptions = positioningOption.updatePopperOptions || noop;\n\t\t\t\tlet popperOptions = updatePopperOptions(getPopperOptions(positioningOption, rtl));\n\t\t\t\tpopperInstance.setOptions(popperOptions);\n\t\t\t}\n\t\t},\n\t\tdestroy() {\n\t\t\tif (popperInstance) {\n\t\t\t\tpopperInstance.destroy();\n\t\t\t\tpopperInstance = null;\n\t\t\t}\n\t\t},\n\t};\n}\n"]}