Skip to content

Commit

Permalink
fixes #52911 #53070 and aria roles with menuitem and submenuitems
Browse files Browse the repository at this point in the history
  • Loading branch information
sbatten committed Jun 30, 2018
1 parent db2f851 commit a7ba364
Show file tree
Hide file tree
Showing 2 changed files with 120 additions and 26 deletions.
7 changes: 7 additions & 0 deletions src/vs/base/browser/ui/menu/menu.css
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,13 @@
background-color: #EEE;
}

.monaco-menu .monaco-action-bar.vertical .action-menu-item {
-ms-flex: 1 1 auto;
flex: 1 1 auto;
display: -ms-flexbox;
display: flex;
}

.monaco-menu .monaco-action-bar.vertical .action-label {
-ms-flex: 1 1 auto;
flex: 1 1 auto;
Expand Down
139 changes: 113 additions & 26 deletions src/vs/base/browser/ui/menu/menu.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,15 @@
'use strict';

import 'vs/css!./menu';
import * as nls from 'vs/nls';
import { IDisposable } from 'vs/base/common/lifecycle';
import { IActionRunner, IAction, Action } from 'vs/base/common/actions';
import { ActionBar, IActionItemProvider, ActionsOrientation, Separator, ActionItem, IActionItemOptions } from 'vs/base/browser/ui/actionbar/actionbar';
import { ActionBar, IActionItemProvider, ActionsOrientation, Separator, ActionItem, IActionItemOptions, BaseActionItem } from 'vs/base/browser/ui/actionbar/actionbar';
import { ResolvedKeybinding, KeyCode } from 'vs/base/common/keyCodes';
import { Event } from 'vs/base/common/event';
import { addClass, EventType, EventHelper, EventLike } from 'vs/base/browser/dom';
import { addClass, EventType, EventHelper, EventLike, removeTabIndexAndUpdateFocus } from 'vs/base/browser/dom';
import { StandardKeyboardEvent } from 'vs/base/browser/keyboardEvent';
import { $ } from 'vs/base/browser/builder';
import { $, Builder } from 'vs/base/browser/builder';

export interface IMenuOptions {
context?: any;
Expand Down Expand Up @@ -65,7 +66,7 @@ export class Menu {
this.actionBar.push(actions, { icon: true, label: true, isMenu: true });
}

private doGetActionItem(action: IAction, options: IMenuOptions, parentData: ISubMenuData): ActionItem {
private doGetActionItem(action: IAction, options: IMenuOptions, parentData: ISubMenuData): BaseActionItem {
if (action instanceof Separator) {
return new ActionItem(options.context, action, { icon: true });
} else if (action instanceof SubmenuAction) {
Expand Down Expand Up @@ -110,42 +111,128 @@ export class Menu {
}
}

class MenuActionItem extends ActionItem {
class MenuActionItem extends BaseActionItem {
static MNEMONIC_REGEX: RegExp = /&&(.)/g;

protected $e: Builder;
protected $label: Builder;
protected options: IActionItemOptions;
private cssClass: string;

constructor(ctx: any, action: IAction, options: IActionItemOptions = {}) {
options.isMenu = true;
super(action, action, options);
}

private _addMnemonic(action: IAction, actionItemElement: HTMLElement): void {
let matches = MenuActionItem.MNEMONIC_REGEX.exec(action.label);
if (matches && matches.length === 2) {
let mnemonic = matches[1];
this.options = options;
this.options.icon = options.icon !== undefined ? options.icon : false;
this.options.label = options.label !== undefined ? options.label : true;
this.cssClass = '';
}

let ariaLabel = action.label.replace(MenuActionItem.MNEMONIC_REGEX, mnemonic);
public render(container: HTMLElement): void {
super.render(container);

actionItemElement.accessKey = mnemonic.toLocaleLowerCase();
this.$e.attr('aria-label', ariaLabel);
this.$e = $('a.action-menu-item').appendTo(this.builder);
if (this._action.id === Separator.ID) {
// A separator is a presentation item
this.$e.attr({ role: 'presentation' });
} else {
this.$e.attr('aria-label', action.label);
this.$e.attr({ role: 'menuitem' });
}
}

public render(container: HTMLElement): void {
super.render(container);
this.$label = $('span.action-label').appendTo(this.$e);

this._addMnemonic(this.getAction(), container);
this.$e.attr('role', 'menuitem');
if (this.options.label && this.options.keybinding) {
$('span.keybinding').text(this.options.keybinding).appendTo(this.$e);
}

this._updateClass();
this._updateLabel();
this._updateTooltip();
this._updateEnabled();
this._updateChecked();
}

public focus(): void {
super.focus();
this.$e.domFocus();
}

public _updateLabel(): void {
if (this.options.label) {
let label = this.getAction().label;
if (label && this.options.isMenu) {
if (label) {
let matches = MenuActionItem.MNEMONIC_REGEX.exec(label);
if (matches && matches.length === 2) {
let mnemonic = matches[1];

let ariaLabel = label.replace(MenuActionItem.MNEMONIC_REGEX, mnemonic);

this.$e.getHTMLElement().accessKey = mnemonic.toLocaleLowerCase();
this.$label.attr('aria-label', ariaLabel);
} else {
this.$label.attr('aria-label', label);
}

label = label.replace(MenuActionItem.MNEMONIC_REGEX, '$1\u0332');
}
this.$e.text(label);

this.$label.text(label);
}
}

public _updateTooltip(): void {
let title: string = null;

if (this.getAction().tooltip) {
title = this.getAction().tooltip;

} else if (!this.options.label && this.getAction().label && this.options.icon) {
title = this.getAction().label;

if (this.options.keybinding) {
title = nls.localize({ key: 'titleLabel', comment: ['action title', 'action keybinding'] }, "{0} ({1})", title, this.options.keybinding);
}
}

if (title) {
this.$e.attr({ title: title });
}
}

public _updateClass(): void {
if (this.cssClass) {
this.$e.removeClass(this.cssClass);
}
if (this.options.icon) {
this.cssClass = this.getAction().class;
this.$label.addClass('icon');
if (this.cssClass) {
this.$label.addClass(this.cssClass);
}
this._updateEnabled();
} else {
this.$label.removeClass('icon');
}
}

public _updateEnabled(): void {
if (this.getAction().enabled) {
this.builder.removeClass('disabled');
this.$e.removeClass('disabled');
this.$e.attr({ tabindex: 0 });
} else {
this.builder.addClass('disabled');
this.$e.addClass('disabled');
removeTabIndexAndUpdateFocus(this.$e.getHTMLElement());
}
}

public _updateChecked(): void {
if (this.getAction().checked) {
this.$label.addClass('checked');
} else {
this.$label.removeClass('checked');
}
}
}
Expand All @@ -166,10 +253,9 @@ class SubmenuActionItem extends MenuActionItem {
public render(container: HTMLElement): void {
super.render(container);

this.builder = $(container);
$(this.builder).addClass('monaco-submenu-item');
$('span.submenu-indicator').text('\u25B6').appendTo(this.builder);
this.$e.attr('role', 'menu');
this.$e.addClass('monaco-submenu-item');
this.$e.attr('aria-haspopup', 'true');
$('span.submenu-indicator').text('\u25B6').appendTo(this.$e);

$(this.builder).on(EventType.KEY_UP, (e) => {
let event = new StandardKeyboardEvent(e as KeyboardEvent);
Expand Down Expand Up @@ -201,7 +287,6 @@ class SubmenuActionItem extends MenuActionItem {
}
});


$(this.builder).on(EventType.MOUSE_LEAVE, (e) => {
this.mouseOver = false;

Expand All @@ -218,6 +303,8 @@ class SubmenuActionItem extends MenuActionItem {
public onClick(e: EventLike) {
// stop clicking from trying to run an action
EventHelper.stop(e, true);

this.createSubmenu();
}

private cleanupExistingSubmenu(force: boolean) {
Expand Down

0 comments on commit a7ba364

Please sign in to comment.