Skip to content

Commit afbe887

Browse files
author
Akos Kitta
committed
show custom title bar on macos
Signed-off-by: Akos Kitta <[email protected]>
1 parent 479628b commit afbe887

File tree

7 files changed

+114
-43
lines changed

7 files changed

+114
-43
lines changed

arduino-ide-extension/src/browser/arduino-ide-frontend-module.ts

+3
Original file line numberDiff line numberDiff line change
@@ -349,6 +349,7 @@ import { RenameCloudSketch } from './contributions/rename-cloud-sketch';
349349
import { CreateFeatures } from './create/create-features';
350350
import { NativeImageCache } from './native-image-cache';
351351
import { rebindWindowPreferences } from '../electron-browser/theia/core/electron-window-preferences';
352+
import { rebindCorePreferences } from './theia/core/core-preferences';
352353

353354
export default new ContainerModule((bind, unbind, isBound, rebind) => {
354355
// Commands and toolbar items
@@ -1022,4 +1023,6 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => {
10221023
bind(FrontendApplicationContribution).toService(NativeImageCache);
10231024
// enable `'window.titleBarStyle'` on all supported OSs
10241025
rebindWindowPreferences(bind, rebind);
1026+
// enable `'window.menuBarVisibility'` on all supported OSs
1027+
rebindCorePreferences(bind, rebind);
10251028
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
import {
2+
CorePreferenceContribution,
3+
CorePreferences,
4+
corePreferenceSchema,
5+
createCorePreferences,
6+
} from '@theia/core/lib/browser/core-preferences';
7+
import { PreferenceContribution } from '@theia/core/lib/browser/preferences/preference-contribution';
8+
import { PreferenceService } from '@theia/core/lib/browser/preferences/preference-service';
9+
import { deepClone } from '@theia/core/lib/common/objects';
10+
import { PreferenceSchema } from '@theia/core/lib/common/preferences/preference-schema';
11+
import { interfaces } from '@theia/core/shared/inversify';
12+
13+
const copy = deepClone(corePreferenceSchema);
14+
copy.properties['window.menuBarVisibility'].included = true; // https://github.com/arduino/arduino-ide/issues/1733#issuecomment-1424186676
15+
const corePreferencesSchema: PreferenceSchema = copy;
16+
17+
export function rebindCorePreferences(
18+
bind: interfaces.Bind,
19+
rebind: interfaces.Rebind
20+
): void {
21+
rebind(CorePreferences)
22+
.toDynamicValue((ctx) => {
23+
const preferences =
24+
ctx.container.get<PreferenceService>(PreferenceService);
25+
const contribution = ctx.container.get<PreferenceContribution>(
26+
CorePreferenceContribution
27+
);
28+
return createCorePreferences(preferences, contribution.schema);
29+
})
30+
.inSingletonScope();
31+
rebind(CorePreferenceContribution).toConstantValue({
32+
schema: corePreferencesSchema,
33+
});
34+
}

arduino-ide-extension/src/browser/toolbar/arduino-toolbar-contribution.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -57,8 +57,8 @@ export class ArduinoToolbarContribution
5757
}
5858

5959
onStart(app: FrontendApplication) {
60-
app.shell.addWidget(this.arduinoToolbarContainer, {
61-
area: 'top',
62-
});
60+
// app.shell.addWidget(this.arduinoToolbarContainer, {
61+
// area: 'top',
62+
// });
6363
}
6464
}

arduino-ide-extension/src/electron-browser/theia/core/electron-main-menu-factory.ts

+4-20
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,6 @@ import {
77
CommandMenuNode,
88
CompoundMenuNode,
99
CompoundMenuNodeRole,
10-
MAIN_MENU_BAR,
1110
MenuNode,
1211
MenuPath,
1312
} from '@theia/core/lib/common/menu';
@@ -28,7 +27,7 @@ export class ElectronMainMenuFactory extends TheiaElectronMainMenuFactory {
2827
@inject(FrontendApplicationStateService)
2928
private readonly appStateService: FrontendApplicationStateService;
3029

31-
private appReady = false;
30+
appReady = false;
3231
private updateWhenReady = false;
3332

3433
override postConstruct(): void {
@@ -41,18 +40,9 @@ export class ElectronMainMenuFactory extends TheiaElectronMainMenuFactory {
4140
});
4241
}
4342

44-
override createElectronMenuBar(): Electron.Menu {
43+
override createElectronMenuBar(): Electron.Menu | null {
4544
this._toggledCommands.clear(); // https://github.com/eclipse-theia/theia/issues/8977
46-
const menuModel = this.menuProvider.getMenu(MAIN_MENU_BAR);
47-
const template = this.fillMenuTemplate([], menuModel, [], {
48-
rootMenuPath: MAIN_MENU_BAR,
49-
});
50-
if (isOSX) {
51-
template.unshift(this.createOSXMenu());
52-
}
53-
const menu = remote.Menu.buildFromTemplate(this.escapeAmpersand(template));
54-
this._menu = menu;
55-
return menu;
45+
return super.createElectronMenuBar();
5646
}
5747

5848
override async setMenuBar(): Promise<void> {
@@ -63,13 +53,7 @@ export class ElectronMainMenuFactory extends TheiaElectronMainMenuFactory {
6353
this.updateWhenReady = true;
6454
return;
6555
}
66-
await this.preferencesService.ready;
67-
const createdMenuBar = this.createElectronMenuBar();
68-
if (isOSX) {
69-
remote.Menu.setApplicationMenu(createdMenuBar);
70-
} else {
71-
remote.getCurrentWindow().setMenu(createdMenuBar);
72-
}
56+
return super.setMenuBar();
7357
}
7458

7559
override createElectronContextMenu(

arduino-ide-extension/src/electron-browser/theia/core/electron-menu-contribution.ts

+36-16
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,25 @@
1-
import { injectable } from '@theia/core/shared/inversify';
1+
import {
2+
getCurrentWebContents,
3+
getCurrentWindow,
4+
Menu,
5+
} from '@theia/core/electron-shared/@electron/remote';
6+
import {
7+
BrowserWindow,
8+
Menu as ElectronMenu,
9+
} from '@theia/core/electron-shared/electron';
10+
import { FrontendApplication } from '@theia/core/lib/browser/frontend-application';
11+
import { KeybindingRegistry } from '@theia/core/lib/browser/keybinding';
12+
import { PreferenceScope } from '@theia/core/lib/browser/preferences/preference-scope';
213
import { CommandRegistry } from '@theia/core/lib/common/command';
314
import { MenuModelRegistry } from '@theia/core/lib/common/menu';
4-
import { KeybindingRegistry } from '@theia/core/lib/browser/keybinding';
15+
import { isOSX } from '@theia/core/lib/common/os';
516
import {
6-
ElectronMenuContribution as TheiaElectronMenuContribution,
717
ElectronCommands,
18+
ElectronMenuContribution as TheiaElectronMenuContribution,
819
} from '@theia/core/lib/electron-browser/menu/electron-menu-contribution';
9-
import { MainMenuManager } from '../../../common/main-menu-manager';
10-
import { FrontendApplication } from '@theia/core/lib/browser/frontend-application';
1120
import { ZoomLevel } from '@theia/core/lib/electron-browser/window/electron-window-preferences';
12-
import { PreferenceScope } from '@theia/core/lib/browser/preferences/preference-scope';
13-
import {
14-
getCurrentWindow,
15-
getCurrentWebContents,
16-
} from '@theia/core/electron-shared/@electron/remote';
21+
import { injectable } from '@theia/core/shared/inversify';
22+
import { MainMenuManager } from '../../../common/main-menu-manager';
1723

1824
@injectable()
1925
export class ElectronMenuContribution
@@ -27,14 +33,28 @@ export class ElectronMenuContribution
2733
super.onStart(app);
2834
}
2935

30-
protected override hideTopPanel(): void {
31-
// NOOP
32-
// IDE2 reuses the `div` for the toolbar.
36+
update(): void {
37+
// no menu updates before `onStart`
38+
if (!this.app) {
39+
return;
40+
}
41+
this.setMenu(this.app);
3342
}
3443

35-
update(): void {
36-
if (this.app) {
37-
this.setMenu(this.app);
44+
protected override setMenu(
45+
app: FrontendApplication,
46+
electronMenu: ElectronMenu | null = this.factory.createElectronMenuBar(),
47+
electronWindow: BrowserWindow = getCurrentWindow()
48+
): void {
49+
this.hideTopPanel(app); // Updates the top panel's visibility on all operating systems.
50+
if (this.titleBarStyle === 'custom' && !this.menuBar) {
51+
this.createCustomTitleBar(app, electronWindow);
52+
}
53+
if (isOSX) {
54+
Menu.setApplicationMenu(electronMenu);
55+
} else {
56+
// Unix/Windows: Set the per-window menus
57+
electronWindow.setMenu(electronMenu);
3858
}
3959
}
4060

arduino-ide-extension/src/electron-browser/theia/core/electron-window-preferences.ts

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import { interfaces } from '@theia/core/shared/inversify';
1212

1313
const copy = deepClone(theiaElectronWindowPreferencesSchema);
1414
copy.properties['window.titleBarStyle'].included = true; // https://github.com/eclipse-theia/theia/pull/10044#issuecomment-1423841453
15+
copy.properties['window.titleBarStyle'].default = 'custom';
1516
const electronWindowPreferencesSchema: PreferenceSchema = copy;
1617

1718
export function rebindWindowPreferences(

arduino-ide-extension/src/electron-main/theia/electron-main-application.ts

+33-4
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,12 @@ import {
2020
import { URI } from '@theia/core/shared/vscode-uri';
2121
import { Deferred } from '@theia/core/lib/common/promise-util';
2222
import * as os from '@theia/core/lib/common/os';
23-
import { Restart } from '@theia/core/lib/electron-common/messaging/electron-messages';
23+
import {
24+
RequestTitleBarStyle,
25+
Restart,
26+
TitleBarStyleAtStartup,
27+
TitleBarStyleChanged,
28+
} from '@theia/core/lib/electron-common/messaging/electron-messages';
2429
import { TheiaBrowserWindowOptions } from '@theia/core/lib/electron-main/theia-electron-window';
2530
import { IsTempSketch } from '../../node/is-temp-sketch';
2631
import {
@@ -330,10 +335,19 @@ export class ElectronMainApplication extends TheiaElectronMainApplication {
330335
}
331336

332337
protected override getTitleBarStyle(
333-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
334-
_config: FrontendApplicationConfig
338+
config: FrontendApplicationConfig
335339
): 'native' | 'custom' {
336-
return 'native';
340+
const storedFrame = this.electronStore.get('windowstate')?.frame;
341+
if (storedFrame !== undefined) {
342+
return !!storedFrame ? 'native' : 'custom';
343+
}
344+
if (config.preferences && config.preferences['window.titleBarStyle']) {
345+
const titleBarStyle = config.preferences['window.titleBarStyle'];
346+
if (titleBarStyle === 'native' || titleBarStyle === 'custom') {
347+
return titleBarStyle;
348+
}
349+
}
350+
return 'custom';
337351
}
338352

339353
protected override hookApplicationEvents(): void {
@@ -351,6 +365,21 @@ export class ElectronMainApplication extends TheiaElectronMainApplication {
351365
this.delete(sketch);
352366
}
353367
});
368+
ipcMain.on(TitleBarStyleChanged, ({ sender }, titleBarStyle: string) => {
369+
this.useNativeWindowFrame = titleBarStyle === 'native'; // XXX: No `isOSX || ` check
370+
const browserWindow = BrowserWindow.fromId(sender.id);
371+
if (browserWindow) {
372+
this.saveWindowState(browserWindow);
373+
} else {
374+
console.warn(`no BrowserWindow with id: ${sender.id}`);
375+
}
376+
});
377+
ipcMain.on(RequestTitleBarStyle, ({ sender }) => {
378+
sender.send(
379+
TitleBarStyleAtStartup,
380+
this.didUseNativeWindowFrameOnStart.get(sender.id) ? 'native' : 'custom'
381+
);
382+
});
354383
}
355384

356385
protected override async onSecondInstance(

0 commit comments

Comments
 (0)