-
-
Notifications
You must be signed in to change notification settings - Fork 130
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
Collapse main menu options in a hamburger menu #489
Changes from 1 commit
2f46ffc
59291de
4f9f0e1
dfe9abd
b5402a8
f7501b7
1e1c494
de210ed
4e8c625
c14d733
fcc357b
8cca0f6
aae3aca
36cefff
92fbf60
8598c45
3f0c8e6
76b3125
f1f3879
705a855
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||
---|---|---|---|---|---|---|---|---|
|
@@ -51,6 +51,7 @@ export class MenuBar extends Widget { | |||||||
}; | ||||||||
this._overflowMenu = null; | ||||||||
this._menuItemSizes = []; | ||||||||
this._overflowIndex = -1; | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -187,8 +188,8 @@ export class MenuBar extends Widget { | |||||||
* #### Notes | ||||||||
* If the menu is already added to the menu bar, it will be moved. | ||||||||
*/ | ||||||||
addMenu(menu: Menu): void { | ||||||||
this.insertMenu(this._menus.length, menu); | ||||||||
addMenu(menu: Menu, update: boolean = true): void { | ||||||||
this.insertMenu(this._menus.length, menu, update); | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -203,7 +204,7 @@ export class MenuBar extends Widget { | |||||||
* | ||||||||
* If the menu is already added to the menu bar, it will be moved. | ||||||||
*/ | ||||||||
insertMenu(index: number, menu: Menu): void { | ||||||||
insertMenu(index: number, menu: Menu, update: boolean = true): void { | ||||||||
// Close the child menu before making changes. | ||||||||
this._closeChildMenu(); | ||||||||
|
||||||||
|
@@ -227,7 +228,9 @@ export class MenuBar extends Widget { | |||||||
menu.title.changed.connect(this._onTitleChanged, this); | ||||||||
|
||||||||
// Schedule an update of the items. | ||||||||
this.update(); | ||||||||
if (update) { | ||||||||
this.update(); | ||||||||
} | ||||||||
|
||||||||
// There is nothing more to do. | ||||||||
return; | ||||||||
|
@@ -249,7 +252,9 @@ export class MenuBar extends Widget { | |||||||
ArrayExt.move(this._menus, i, j); | ||||||||
|
||||||||
// Schedule an update of the items. | ||||||||
this.update(); | ||||||||
if (update) { | ||||||||
this.update(); | ||||||||
} | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -260,8 +265,8 @@ export class MenuBar extends Widget { | |||||||
* #### Notes | ||||||||
* This is a no-op if the menu is not in the menu bar. | ||||||||
*/ | ||||||||
removeMenu(menu: Menu): void { | ||||||||
this.removeMenuAt(this._menus.indexOf(menu)); | ||||||||
removeMenu(menu: Menu, update: boolean = true): void { | ||||||||
this.removeMenuAt(this._menus.indexOf(menu), update); | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -272,7 +277,7 @@ export class MenuBar extends Widget { | |||||||
* #### Notes | ||||||||
* This is a no-op if the index is out of range. | ||||||||
*/ | ||||||||
removeMenuAt(index: number): void { | ||||||||
removeMenuAt(index: number, update: boolean = true): void { | ||||||||
// Close the child menu before making changes. | ||||||||
this._closeChildMenu(); | ||||||||
|
||||||||
|
@@ -293,7 +298,9 @@ export class MenuBar extends Widget { | |||||||
menu.removeClass('lm-MenuBar-menu'); | ||||||||
|
||||||||
// Schedule an update of the items. | ||||||||
this.update(); | ||||||||
if (update) { | ||||||||
this.update(); | ||||||||
} | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -351,9 +358,6 @@ export class MenuBar extends Widget { | |||||||
event.preventDefault(); | ||||||||
event.stopPropagation(); | ||||||||
break; | ||||||||
case 'resize': | ||||||||
this._evtResize(event); | ||||||||
break; | ||||||||
} | ||||||||
} | ||||||||
|
||||||||
|
@@ -366,7 +370,6 @@ export class MenuBar extends Widget { | |||||||
this.node.addEventListener('mousemove', this); | ||||||||
this.node.addEventListener('mouseleave', this); | ||||||||
this.node.addEventListener('contextmenu', this); | ||||||||
window.addEventListener('resize', this); | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -378,7 +381,6 @@ export class MenuBar extends Widget { | |||||||
this.node.removeEventListener('mousemove', this); | ||||||||
this.node.removeEventListener('mouseleave', this); | ||||||||
this.node.removeEventListener('contextmenu', this); | ||||||||
window.removeEventListener('resize', this); | ||||||||
this._closeChildMenu(); | ||||||||
} | ||||||||
|
||||||||
|
@@ -392,9 +394,13 @@ export class MenuBar extends Widget { | |||||||
} | ||||||||
|
||||||||
/** | ||||||||
* A message handler invoked on an `'resize'` message. | ||||||||
* A message handler invoked on a `'resize'` message. | ||||||||
*/ | ||||||||
protected _evtResize(event: Event): void { | ||||||||
protected onResize(msg: Widget.ResizeMessage): void { | ||||||||
this.update(); | ||||||||
fcollonval marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||
} | ||||||||
|
||||||||
protected updateOverflowIndex(): void { | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do we need to expose this method, or could it be private? If it should be protected to allow overriding, maybe it could return a new (private) interface, like: interface IMenuOverflowState {
itemSizes: number[];
index: number
} then you can have a single private variable set as: this._overflowState = this.calculateOverflowState(); and use it with unpacking: if (!this._overflowState) {
return; // or otherwise skip
}
const { index: overflowIndex, itemSizes: menuItemSizes } = this._overflowState; Then it would be also easier to unit-test it too. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I tried to create the interface, but soon enough I found myself wondering that I needed to update the overflow index value in multiple spots, so I'm not sure if we need the interface. For now I just converted the function to private. |
||||||||
// Get elements visible in the main menu bar | ||||||||
let itemMenus = this.node.getElementsByClassName('lm-MenuBar-item'); | ||||||||
steff456 marked this conversation as resolved.
Show resolved
Hide resolved
steff456 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||
let screenSize = this.node.offsetWidth; | ||||||||
|
@@ -423,39 +429,7 @@ export class MenuBar extends Widget { | |||||||
} | ||||||||
} | ||||||||
} | ||||||||
|
||||||||
if (index > -1) { | ||||||||
// Create hamburger menu | ||||||||
if (this._overflowMenu === null) { | ||||||||
this._overflowMenu = new Menu({ commands: new CommandRegistry() }); | ||||||||
this._overflowMenu.title.label = '...'; | ||||||||
this._overflowMenu.title.mnemonic = 0; | ||||||||
this.addMenu(this._overflowMenu); | ||||||||
} | ||||||||
|
||||||||
// Move menus | ||||||||
for (let i = index; i < n - 1; i++) { | ||||||||
let submenu = this.menus[i]; | ||||||||
submenu.title.mnemonic = 0; | ||||||||
this._overflowMenu.insertItem(0, { | ||||||||
type: 'submenu', | ||||||||
submenu: submenu | ||||||||
}); | ||||||||
this.removeMenuAt(i); | ||||||||
} | ||||||||
} else if (this._overflowMenu !== null) { | ||||||||
let i = n - 1; | ||||||||
let hamburgerMenuItems = this._overflowMenu.items; | ||||||||
if (screenSize - totalMenuSize > this._menuItemSizes[i]) { | ||||||||
let menu = hamburgerMenuItems[0].submenu as Menu; | ||||||||
this._overflowMenu.removeItemAt(0); | ||||||||
this.insertMenu(i, menu); | ||||||||
} | ||||||||
if (this._overflowMenu.items.length === 0) { | ||||||||
this.removeMenu(this._overflowMenu); | ||||||||
this._overflowMenu = null; | ||||||||
} | ||||||||
} | ||||||||
this._overflowIndex = index; | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -465,8 +439,15 @@ export class MenuBar extends Widget { | |||||||
let menus = this._menus; | ||||||||
let renderer = this.renderer; | ||||||||
let activeIndex = this._activeIndex; | ||||||||
let content = new Array<VirtualElement>(menus.length); | ||||||||
for (let i = 0, n = menus.length; i < n; ++i) { | ||||||||
let length = this._overflowIndex > -1 ? this._overflowIndex : menus.length; | ||||||||
let content = new Array<VirtualElement>(length); | ||||||||
let totalMenuSize = 0; | ||||||||
|
||||||||
// Check that the overflow menu doesn't count | ||||||||
length = this._overflowMenu !== null ? length - 1 : length; | ||||||||
|
||||||||
// Render visible menus | ||||||||
for (let i = 0; i < length; ++i) { | ||||||||
let title = menus[i].title; | ||||||||
let active = i === activeIndex; | ||||||||
if (active && menus[i].items.length == 0) { | ||||||||
|
@@ -479,9 +460,72 @@ export class MenuBar extends Widget { | |||||||
this.activeIndex = i; | ||||||||
} | ||||||||
}); | ||||||||
// Calculate size of current menu | ||||||||
totalMenuSize += this._menuItemSizes[i]; | ||||||||
} | ||||||||
// Render overflow menu if needed | ||||||||
if (this._overflowIndex > -1) { | ||||||||
// Create overflow menu | ||||||||
if (this._overflowMenu === null) { | ||||||||
this._overflowMenu = new Menu({ commands: new CommandRegistry() }); | ||||||||
this._overflowMenu.title.label = '...'; | ||||||||
this._overflowMenu.title.mnemonic = 0; | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe worth adding a new optional method in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I see that 4e8c625 added I meant to suggest that we should wrap rendering of the overflow menu so that the
Suggested change
This is very minor, I don't think we have to do this. To limit the surface of API changes I would also suggest reverting There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think @krassowski's suggestion to remove |
||||||||
this.addMenu(this._overflowMenu, false); | ||||||||
} | ||||||||
// Move menus to overflow menu | ||||||||
for (let i = length; i < menus.length - 1; ++i) { | ||||||||
let submenu = this.menus[i]; | ||||||||
steff456 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||
submenu.title.mnemonic = 0; | ||||||||
this._overflowMenu.insertItem(0, { | ||||||||
type: 'submenu', | ||||||||
submenu: submenu | ||||||||
}); | ||||||||
this.removeMenu(submenu, false); | ||||||||
} | ||||||||
let title = this._overflowMenu.title; | ||||||||
let active = length === activeIndex; | ||||||||
if (active && menus[length].items.length == 0) { | ||||||||
active = false; | ||||||||
} | ||||||||
content[length] = renderer.renderItem({ | ||||||||
title, | ||||||||
active, | ||||||||
steff456 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||
onfocus: () => { | ||||||||
this.activeIndex = length; | ||||||||
} | ||||||||
}); | ||||||||
} else if (this._overflowMenu !== null) { | ||||||||
// Remove submenus from overflow menu | ||||||||
let overflowMenuItems = this._overflowMenu.items; | ||||||||
let screenSize = this.node.offsetWidth; | ||||||||
let n = this._overflowMenu.items.length; | ||||||||
for (let i = 0; i < n; ++i) { | ||||||||
let index = menus.length - 1 - i; | ||||||||
if (screenSize - totalMenuSize > this._menuItemSizes[index]) { | ||||||||
let menu = overflowMenuItems[0].submenu as Menu; | ||||||||
this._overflowMenu.removeItemAt(0); | ||||||||
this.insertMenu(length, menu, false); | ||||||||
let title = menu.title; | ||||||||
let active = false; | ||||||||
content[length] = renderer.renderItem({ | ||||||||
title, | ||||||||
active, | ||||||||
steff456 marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||
onfocus: () => { | ||||||||
this.activeIndex = length; | ||||||||
} | ||||||||
}); | ||||||||
length++; | ||||||||
} | ||||||||
} | ||||||||
if (this._overflowMenu.items.length === 0) { | ||||||||
this.removeMenu(this._overflowMenu, false); | ||||||||
content.pop(); | ||||||||
this._overflowMenu = null; | ||||||||
this._overflowIndex = -1; | ||||||||
} | ||||||||
} | ||||||||
this._menuItemSizes = []; | ||||||||
VirtualDOM.render(content, this.contentNode); | ||||||||
this.updateOverflowIndex(); | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It looks like this one will lead to the state being always delayed by one tick. It is not bad in itself if this is a conscious decision made as a performance-UX tradeoff. Could we document this? One thing to consider is whether we should wrap it in an extra There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How do I wrap it in a There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It would be something like: requestAnimationFrame(() => this._updateOverflowIndex()); I have not tested if this helps. You would need to record a profile in developer tools when you resize the page to trigger collapsing before and after applying the change and compare. If you have the profiles it should be obvious from the time spend on forced style recalculation/layout whether it is worth waiting for the next frame or not. |
||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
@@ -809,6 +853,7 @@ export class MenuBar extends Widget { | |||||||
private _childMenu: Menu | null = null; | ||||||||
private _overflowMenu: Menu | null = null; | ||||||||
private _menuItemSizes: number[] = []; | ||||||||
private _overflowIndex: number; | ||||||||
} | ||||||||
|
||||||||
/** | ||||||||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi @krassowski, I remembered that we added this optional argument here in Lumino just in case other applications didn't want to have the collapse behavior. I can check if this argument is set to True in Jupyter, and that should be the solution of not seeing this feature active