Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
  • Loading branch information
ghiscoding committed Oct 28, 2023
2 parents b5576de + 338ea2a commit 7deb170
Show file tree
Hide file tree
Showing 10 changed files with 85 additions and 50 deletions.
2 changes: 1 addition & 1 deletion src/controls/slick.columnmenu.ts
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,7 @@ export class SlickColumnMenu {
const spanCloseElm = document.createElement('span');
spanCloseElm.className = 'close';
spanCloseElm.ariaHidden = 'true';
spanCloseElm.innerHTML = '×';
spanCloseElm.textContent = '×';
buttonElm.appendChild(spanCloseElm);
this._menuElm.appendChild(buttonElm);

Expand Down
2 changes: 1 addition & 1 deletion src/controls/slick.columnpicker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,7 @@ export class SlickColumnPicker {
const spanCloseElm = document.createElement('span');
spanCloseElm.className = 'close';
spanCloseElm.ariaHidden = 'true';
spanCloseElm.innerHTML = '×';
spanCloseElm.textContent = '×';
buttonElm.appendChild(spanCloseElm);
this._menuElm.appendChild(buttonElm);

Expand Down
11 changes: 7 additions & 4 deletions src/controls/slick.gridmenu.ts
Original file line number Diff line number Diff line change
Expand Up @@ -315,7 +315,7 @@ export class SlickGridMenu {
const spanCloseElm = document.createElement('span');
spanCloseElm.className = 'close';
spanCloseElm.ariaHidden = 'true';
spanCloseElm.innerHTML = '×';
spanCloseElm.textContent = '×';
closeButtonElm.appendChild(spanCloseElm);
menuElm.appendChild(closeButtonElm);
}
Expand Down Expand Up @@ -380,14 +380,16 @@ export class SlickGridMenu {

/** Close and destroy all previously opened sub-menus */
destroySubMenus() {
this._bindingEventService.unbindAll('sub-menu');
document.querySelectorAll(`.slick-gridmenu.slick-submenu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}

/** Construct the custom command menu items. */
protected populateCommandsMenu(commandItems: Array<GridMenuItem | MenuCommandItem | 'divider'>, commandListElm: HTMLElement, args: { grid: SlickGrid, level: number }) {
// user could pass a title on top of the custom section
const isSubMenu = args.level > 0;
const level = args?.level || 0;
const isSubMenu = level > 0;
if (!isSubMenu && (this._gridMenuOptions?.commandTitle || this._gridMenuOptions?.customTitle)) {
this._commandTitleElm = document.createElement('div');
this._commandTitleElm.className = 'title';
Expand Down Expand Up @@ -470,14 +472,15 @@ export class SlickGridMenu {
commandListElm.appendChild(liElm);

if (addClickListener) {
this._bindingEventService.bind(liElm, 'click', this.handleMenuItemClick.bind(this, item, args.level) as EventListener);
const eventGroup = isSubMenu ? 'sub-menu' : 'parent-menu';
this._bindingEventService.bind(liElm, 'click', this.handleMenuItemClick.bind(this, item, level) as EventListener, undefined, eventGroup);
}

// optionally open sub-menu(s) by mouseover
if (this._gridMenuOptions?.subMenuOpenByEvent === 'mouseover') {
this._bindingEventService.bind(liElm, 'mouseover', ((e: DOMMouseOrTouchEvent<HTMLDivElement>) => {
if ((item as GridMenuItem).commandItems || (item as GridMenuItem).customItems) {
this.repositionSubMenu(item, args.level, e);
this.repositionSubMenu(item, level, e);
} else if (!isSubMenu) {
this.destroySubMenus();
}
Expand Down
2 changes: 1 addition & 1 deletion src/controls/slick.pager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@ export class SlickGridPager {
destroy() {
this.setPageSize(0);
this._bindingEventService.unbindAll();
this._container.innerHTML = '';
Utils.emptyElement(this._container);
}

protected getNavState() {
Expand Down
1 change: 1 addition & 0 deletions src/models/core.interface.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ export interface ElementEventListener {
element: Element | Window;
eventName: string;
listener: EventListenerOrEventListenerObject;
groupName?: string;
}

export interface EditController {
Expand Down
12 changes: 9 additions & 3 deletions src/plugins/slick.cellmenu.ts
Original file line number Diff line number Diff line change
Expand Up @@ -434,12 +434,16 @@ export class SlickCellMenu implements SlickPlugin {
/** Destroy all parent menus and any sub-menus */
destroyAllMenus() {
this.destroySubMenus();

// remove all parent menu listeners before removing them from the DOM
this._bindingEventService.unbindAll('parent-menu');
document.querySelectorAll(`.slick-cell-menu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}

/** Close and destroy all previously opened sub-menus */
destroySubMenus() {
this._bindingEventService.unbindAll('sub-menu');
document.querySelectorAll(`.slick-cell-menu.slick-submenu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}
Expand Down Expand Up @@ -623,7 +627,8 @@ export class SlickCellMenu implements SlickPlugin {
}

// user could pass a title on top of the Commands/Options section
const isSubMenu = args.level > 0;
const level = args?.level || 0;
const isSubMenu = level > 0;
if (cellMenu?.[`${itemType}Title`] && !isSubMenu) {
this[`_${itemType}TitleElm`] = document.createElement('div');
this[`_${itemType}TitleElm`]!.className = 'slick-menu-title';
Expand Down Expand Up @@ -703,14 +708,15 @@ export class SlickCellMenu implements SlickPlugin {
commandOrOptionMenuElm.appendChild(liElm);

if (addClickListener) {
this._bindingEventService.bind(liElm, 'click', this.handleMenuItemClick.bind(this, item, itemType, args.level) as EventListener);
const eventGroup = isSubMenu ? 'sub-menu' : 'parent-menu';
this._bindingEventService.bind(liElm, 'click', this.handleMenuItemClick.bind(this, item, itemType, level) as EventListener, undefined, eventGroup);
}

// optionally open sub-menu(s) by mouseover
if (this._cellMenuProperties.subMenuOpenByEvent === 'mouseover') {
this._bindingEventService.bind(liElm, 'mouseover', ((e: DOMMouseOrTouchEvent<HTMLDivElement>) => {
if ((item as MenuCommandItem).commandItems || (item as MenuOptionItem).optionItems) {
this.repositionSubMenu(item, itemType, args.level, e);
this.repositionSubMenu(item, itemType, level, e);
this._lastMenuTypeClicked = itemType;
} else if (!isSubMenu) {
this.destroySubMenus();
Expand Down
14 changes: 10 additions & 4 deletions src/plugins/slick.contextmenu.ts
Original file line number Diff line number Diff line change
Expand Up @@ -348,7 +348,7 @@ export class SlickContextMenu implements SlickPlugin {
const spanCloseElm = document.createElement('span');
spanCloseElm.className = 'close';
spanCloseElm.ariaHidden = 'true';
spanCloseElm.innerHTML = '&times;';
spanCloseElm.textContent = '×';
closeButtonElm.appendChild(spanCloseElm);
}

Expand Down Expand Up @@ -450,12 +450,16 @@ export class SlickContextMenu implements SlickPlugin {
/** Destroy all parent menus and any sub-menus */
destroyAllMenus() {
this.destroySubMenus();

// remove all parent menu listeners before removing them from the DOM
this._bindingEventService.unbindAll('parent-menu');
document.querySelectorAll(`.slick-context-menu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}

/** Close and destroy all previously opened sub-menus */
destroySubMenus() {
this._bindingEventService.unbindAll('sub-menu');
document.querySelectorAll(`.slick-context-menu.slick-submenu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}
Expand Down Expand Up @@ -551,7 +555,8 @@ export class SlickContextMenu implements SlickPlugin {
}

// user could pass a title on top of the Commands/Options section
const isSubMenu = args.level > 0;
const level = args?.level || 0;
const isSubMenu = level > 0;
if (contextMenu?.[`${itemType}Title`] && !isSubMenu) {
this[`_${itemType}TitleElm`] = document.createElement('div');
this[`_${itemType}TitleElm`]!.className = 'slick-menu-title';
Expand Down Expand Up @@ -631,14 +636,15 @@ export class SlickContextMenu implements SlickPlugin {
commandOrOptionMenuElm.appendChild(liElm);

if (addClickListener) {
this._bindingEventService.bind(liElm, 'click', this.handleMenuItemClick.bind(this, item, itemType, args.level) as EventListener);
const eventGroup = isSubMenu ? 'sub-menu' : 'parent-menu';
this._bindingEventService.bind(liElm, 'click', this.handleMenuItemClick.bind(this, item, itemType, level) as EventListener, undefined, eventGroup);
}

// optionally open sub-menu(s) by mouseover
if (this._contextMenuProperties.subMenuOpenByEvent === 'mouseover') {
this._bindingEventService.bind(liElm, 'mouseover', ((e: DOMMouseOrTouchEvent<HTMLDivElement>) => {
if ((item as MenuCommandItem).commandItems || (item as MenuOptionItem).optionItems) {
this.repositionSubMenu(item, itemType, args.level, e);
this.repositionSubMenu(item, itemType, level, e);
this._lastMenuTypeClicked = itemType;
} else if (!isSubMenu) {
this.destroySubMenus();
Expand Down
7 changes: 6 additions & 1 deletion src/plugins/slick.headermenu.ts
Original file line number Diff line number Diff line change
Expand Up @@ -171,12 +171,16 @@ export class SlickHeaderMenu implements SlickPlugin {

destroyAllMenus() {
this.destroySubMenus();

// remove all parent menu listeners before removing them from the DOM
this._bindingEventService.unbindAll('parent-menu');
document.querySelectorAll(`.slick-header-menu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}

/** Close and destroy all previously opened sub-menus */
destroySubMenus() {
this._bindingEventService.unbindAll('sub-menu');
document.querySelectorAll(`.slick-header-menu.slick-submenu${this.getGridUidSelector()}`)
.forEach(subElm => subElm.remove());
}
Expand Down Expand Up @@ -425,7 +429,8 @@ export class SlickHeaderMenu implements SlickPlugin {
menuElm.appendChild(menuItemElm);

if (addClickListener) {
this._bindingEventService.bind(menuItemElm, 'click', this.handleMenuItemClick.bind(this, item, columnDef, level) as EventListener);
const eventGroup = isSubMenu ? 'sub-menu' : 'parent-menu';
this._bindingEventService.bind(menuItemElm, 'click', this.handleMenuItemClick.bind(this, item, columnDef, level) as EventListener, undefined, eventGroup);
}

// optionally open sub-menu(s) by mouseover
Expand Down
33 changes: 26 additions & 7 deletions src/slick.core.ts
Original file line number Diff line number Diff line change
Expand Up @@ -552,9 +552,9 @@ export class BindingEventService {
}

/** Bind an event listener to any element */
bind(element: Element | Window, eventName: string, listener: EventListenerOrEventListenerObject, options?: boolean | AddEventListenerOptions) {
bind(element: Element | Window, eventName: string, listener: EventListenerOrEventListenerObject, options?: boolean | AddEventListenerOptions, groupName = '') {
element.addEventListener(eventName, listener, options);
this._boundedEvents.push({ element, eventName, listener });
this._boundedEvents.push({ element, eventName, listener, groupName });
}

/** Unbind all will remove every every event handlers that were bounded earlier */
Expand All @@ -571,11 +571,30 @@ export class BindingEventService {
}
}

/** Unbind all will remove every every event handlers that were bounded earlier */
unbindAll() {
while (this._boundedEvents.length > 0) {
const boundedEvent = this._boundedEvents.pop() as ElementEventListener;
this.unbind(boundedEvent.element, boundedEvent.eventName, boundedEvent.listener);
/**
* Unbind all event listeners that were bounded, optionally provide a group name to unbind all listeners assigned to that specific group only.
*/
unbindAll(groupName?: string | string[]) {
if (groupName) {
const groupNames = Array.isArray(groupName) ? groupName : [groupName];

// unbind only the bounded event with a specific group
// Note: we need to loop in reverse order to avoid array reindexing (causing index offset) after a splice is called
for (let i = this._boundedEvents.length - 1; i >= 0; --i) {
const boundedEvent = this._boundedEvents[i];
if (groupNames.some(g => g === boundedEvent.groupName)) {
const { element, eventName, listener } = boundedEvent;
this.unbind(element, eventName, listener);
this._boundedEvents.splice(i, 1);
}
}
} else {
// unbind everything
while (this._boundedEvents.length > 0) {
const boundedEvent = this._boundedEvents.pop() as ElementEventListener;
const { element, eventName, listener } = boundedEvent;
this.unbind(element, eventName, listener);
}
}
}
}
Expand Down
51 changes: 23 additions & 28 deletions src/slick.grid.ts
Original file line number Diff line number Diff line change
Expand Up @@ -350,7 +350,7 @@ export class SlickGrid<TData = any, C extends Column<TData> = Column<TData>, O e
protected _topPanels!: HTMLDivElement[];
protected _viewport!: HTMLDivElement[];
protected _canvas!: HTMLDivElement[];
protected _style: any;
protected _style?: HTMLStyleElement;
protected _boundAncestors: HTMLElement[] = [];
protected stylesheet?: { cssRules: Array<{ selectorText: string; }>; rules: Array<{ selectorText: string; }>; } | null;
protected columnCssRulesL?: Array<{ selectorText: string; }>;
Expand Down Expand Up @@ -2299,33 +2299,28 @@ export class SlickGrid<TData = any, C extends Column<TData> = Column<TData>, O e
}

protected createCssRules() {
const template = Utils.createDomElement('template', { innerHTML: '<style type="text/css" rel="stylesheet" />' });
this._style = template.content.firstChild;
this._style = document.createElement('style');
this._style.nonce = 'random-string';
document.head.appendChild(this._style);

const rowHeight = (this._options.rowHeight! - this.cellHeightDiff);
const rules = [
`.${this.uid} .slick-group-header-column { left: 1000px; }`,
`.${this.uid} .slick-header-column { left: 1000px; }`,
`.${this.uid} .slick-top-panel { height: ${this._options.topPanelHeight}px; }`,
`.${this.uid} .slick-preheader-panel { height: ${this._options.preHeaderPanelHeight}px; }`,
`.${this.uid} .slick-headerrow-columns { height: ${this._options.headerRowHeight}px; }`,
`.${this.uid} .slick-footerrow-columns { height: ${this._options.footerRowHeight}px; }`,
`.${this.uid} .slick-cell { height: ${rowHeight}px; }`,
`.${this.uid} .slick-row { height: ${this._options.rowHeight}px; }`
];
const sheet = this._style.sheet;
if (sheet) {
const rowHeight = (this._options.rowHeight! - this.cellHeightDiff);
sheet.insertRule(`.${this.uid} .slick-group-header-column { left: 1000px; }`);
sheet.insertRule(`.${this.uid} .slick-header-column { left: 1000px; }`);
sheet.insertRule(`.${this.uid} .slick-top-panel { height: ${this._options.topPanelHeight}px; }`);
sheet.insertRule(`.${this.uid} .slick-preheader-panel { height: ${this._options.preHeaderPanelHeight}px; }`);
sheet.insertRule(`.${this.uid} .slick-headerrow-columns { height: ${this._options.headerRowHeight}px; }`);
sheet.insertRule(`.${this.uid} .slick-footerrow-columns { height: ${this._options.footerRowHeight}px; }`);
sheet.insertRule(`.${this.uid} .slick-cell { height: ${rowHeight}px; }`);
sheet.insertRule(`.${this.uid} .slick-row { height: ${this._options.rowHeight}px; }`);

for (let i = 0; i < this.columns.length; i++) {
if (!this.columns[i] || this.columns[i].hidden) { continue; }

rules.push(`.${this.uid} .l${i} { }`);
rules.push(`.${this.uid} .r${i} { }`);
}
for (let i = 0; i < this.columns.length; i++) {
if (!this.columns[i] || this.columns[i].hidden) { continue; }

if (this._style.styleSheet) { // IE
this._style.styleSheet.cssText = rules.join(' ');
} else {
this._style.appendChild(document.createTextNode(rules.join(' ')));
sheet.insertRule(`.${this.uid} .l${i} { }`);
sheet.insertRule(`.${this.uid} .r${i} { }`);
}
}
}

Expand Down Expand Up @@ -2369,7 +2364,7 @@ export class SlickGrid<TData = any, C extends Column<TData> = Column<TData>, O e
}

protected removeCssRules() {
this._style.remove();
this._style?.remove();
this.stylesheet = null;
}

Expand Down Expand Up @@ -2976,7 +2971,7 @@ export class SlickGrid<TData = any, C extends Column<TData> = Column<TData>, O e
}
}

cellEl.innerHTML = maxText;
cellEl.textContent = maxText;
len = cellEl.offsetWidth;

rowEl.remove();
Expand Down Expand Up @@ -4090,7 +4085,7 @@ export class SlickGrid<TData = any, C extends Column<TData> = Column<TData>, O e
formatterResult = this.getFormatter(row, m)(row, columnIdx, this.getDataItemValueForColumn(d, m), m, d, this as unknown as SlickGridModel);
this.applyFormatResultToCellNode(formatterResult, node as HTMLDivElement);
} else {
node.innerHTML = '';
Utils.emptyElement(node);
}
}

Expand Down Expand Up @@ -5771,7 +5766,7 @@ export class SlickGrid<TData = any, C extends Column<TData> = Column<TData>, O e

// don't clear the cell if a custom editor is passed through
if (!editor && !useEditor.suppressClearOnEdit) {
this.activeCellNode.innerHTML = '';
Utils.emptyElement(this.activeCellNode);
}

let metadata = (this.data as CustomDataView<TData>)?.getItemMetadata?.(this.activeRow);
Expand Down

0 comments on commit 7deb170

Please sign in to comment.