Merge pull request #798 from mathuo/797-popout-group-flow-error

bug: fixup popout group flows
This commit is contained in:
mathuo 2024-12-21 19:24:50 +00:00 committed by GitHub
commit 6fbe2b4233
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 349 additions and 104 deletions

View File

@ -140,109 +140,114 @@ describe('dockviewComponent', () => {
expect(dockview.element.className).toBe('test-b test-c'); expect(dockview.element.className).toBe('test-b test-c');
}); });
// describe('memory leakage', () => { describe('memory leakage', () => {
// beforeEach(() => { beforeEach(() => {
// window.open = () => fromPartial<Window>({ window.open = () => setupMockWindow();
// addEventListener: jest.fn(), });
// close: jest.fn(),
// });
// });
// test('event leakage', () => { test('event leakage', async () => {
// Emitter.setLeakageMonitorEnabled(true); Emitter.setLeakageMonitorEnabled(true);
// dockview = new DockviewComponent({ dockview = new DockviewComponent(container, {
// parentElement: container, createComponent(options) {
// components: { switch (options.name) {
// default: PanelContentPartTest, case 'default':
// }, return new PanelContentPartTest(
// }); options.id,
options.name
);
default:
throw new Error(`unsupported`);
}
},
className: 'test-a test-b',
});
// dockview.layout(500, 1000); dockview.layout(500, 1000);
// const panel1 = dockview.addPanel({ const panel1 = dockview.addPanel({
// id: 'panel1', id: 'panel1',
// component: 'default', component: 'default',
// }); });
// const panel2 = dockview.addPanel({ const panel2 = dockview.addPanel({
// id: 'panel2', id: 'panel2',
// component: 'default', component: 'default',
// }); });
// dockview.removePanel(panel2); dockview.removePanel(panel2);
// const panel3 = dockview.addPanel({ const panel3 = dockview.addPanel({
// id: 'panel3', id: 'panel3',
// component: 'default', component: 'default',
// position: { position: {
// direction: 'right', direction: 'right',
// referencePanel: 'panel1', referencePanel: 'panel1',
// }, },
// }); });
// const panel4 = dockview.addPanel({ const panel4 = dockview.addPanel({
// id: 'panel4', id: 'panel4',
// component: 'default', component: 'default',
// position: { position: {
// direction: 'above', direction: 'above',
// }, },
// }); });
// dockview.moveGroupOrPanel( panel4.api.group.api.moveTo({
// panel4.group, group: panel3.api.group,
// panel3.group.id, position: 'center',
// panel3.id, });
// 'center'
// );
// dockview.addPanel({ dockview.addPanel({
// id: 'panel5', id: 'panel5',
// component: 'default', component: 'default',
// floating: true, floating: true,
// }); });
// const panel6 = dockview.addPanel({ const panel6 = dockview.addPanel({
// id: 'panel6', id: 'panel6',
// component: 'default', component: 'default',
// position: { position: {
// referencePanel: 'panel5', referencePanel: 'panel5',
// direction: 'within', direction: 'within',
// }, },
// }); });
// dockview.addFloatingGroup(panel4.api.group); dockview.addFloatingGroup(panel4.api.group);
// dockview.addPopoutGroup(panel6); await dockview.addPopoutGroup(panel2);
// dockview.moveGroupOrPanel( panel1.api.group.api.moveTo({
// panel1.group, group: panel6.api.group,
// panel6.group.id, position: 'center',
// panel6.id, });
// 'center'
// );
// dockview.moveGroupOrPanel( panel4.api.group.api.moveTo({
// panel4.group, group: panel6.api.group,
// panel6.group.id, position: 'center',
// panel6.id, });
// 'center'
// );
// dockview.dispose(); dockview.dispose();
// if (Emitter.MEMORY_LEAK_WATCHER.size > 0) { if (Emitter.MEMORY_LEAK_WATCHER.size > 0) {
// for (const entry of Array.from( console.warn(
// Emitter.MEMORY_LEAK_WATCHER.events `${Emitter.MEMORY_LEAK_WATCHER.size} undisposed resources`
// )) { );
// console.log('disposal', entry[1]);
// }
// throw new Error('not all listeners disposed');
// }
// Emitter.setLeakageMonitorEnabled(false); for (const entry of Array.from(
// }); Emitter.MEMORY_LEAK_WATCHER.events
// }); )) {
console.log('disposal', entry[1]);
}
throw new Error(
`${Emitter.MEMORY_LEAK_WATCHER.size} undisposed resources`
);
}
Emitter.setLeakageMonitorEnabled(false);
});
});
test('duplicate panel', () => { test('duplicate panel', () => {
dockview.layout(500, 1000); dockview.layout(500, 1000);
@ -3707,16 +3712,16 @@ describe('dockviewComponent', () => {
floatingGroups: [ floatingGroups: [
{ {
data: { data: {
views: ['panelB'], views: ['panelC'],
activeView: 'panelB', activeView: 'panelC',
id: '3', id: '3',
}, },
position: { left: 0, top: 0, height: 100, width: 100 }, position: { left: 0, top: 0, height: 100, width: 100 },
}, },
{ {
data: { data: {
views: ['panelC'], views: ['panelD'],
activeView: 'panelC', activeView: 'panelD',
id: '4', id: '4',
}, },
position: { left: 0, top: 0, height: 100, width: 100 }, position: { left: 0, top: 0, height: 100, width: 100 },
@ -4867,6 +4872,155 @@ describe('dockviewComponent', () => {
); );
}); });
test('basic', async () => {
jest.useRealTimers();
const container = document.createElement('div');
window.open = () => setupMockWindow();
const dockview = new DockviewComponent(container, {
createComponent(options) {
switch (options.name) {
case 'default':
return new PanelContentPartTest(
options.id,
options.name
);
default:
throw new Error(`unsupported`);
}
},
});
dockview.layout(1000, 1000);
dockview.fromJSON({
activeGroup: 'group-1',
grid: {
root: {
type: 'branch',
data: [
{
type: 'leaf',
data: {
views: ['panel1'],
id: 'group-1',
activeView: 'panel1',
},
size: 1000,
},
],
size: 1000,
},
height: 1000,
width: 1000,
orientation: Orientation.VERTICAL,
},
popoutGroups: [
{
data: {
views: ['panel2'],
id: 'group-2',
activeView: 'panel2',
},
position: null,
},
],
panels: {
panel1: {
id: 'panel1',
contentComponent: 'default',
title: 'panel1',
},
panel2: {
id: 'panel2',
contentComponent: 'default',
title: 'panel2',
},
},
});
await new Promise((resolve) => setTimeout(resolve, 0));
const panel2 = dockview.api.getPanel('panel2');
const windowObject =
panel2?.api.location.type === 'popout'
? panel2?.api.location.getWindow()
: undefined;
expect(windowObject).toBeTruthy();
windowObject!.close();
});
test('grid -> floating -> popout -> popout closed', async () => {
const container = document.createElement('div');
window.open = () => setupMockWindow();
const dockview = new DockviewComponent(container, {
createComponent(options) {
switch (options.name) {
case 'default':
return new PanelContentPartTest(
options.id,
options.name
);
default:
throw new Error(`unsupported`);
}
},
});
dockview.layout(1000, 500);
const panel1 = dockview.addPanel({
id: 'panel_1',
component: 'default',
});
const panel2 = dockview.addPanel({
id: 'panel_2',
component: 'default',
});
const panel3 = dockview.addPanel({
id: 'panel_3',
component: 'default',
position: { direction: 'right' },
});
expect(panel1.api.location.type).toBe('grid');
expect(panel2.api.location.type).toBe('grid');
expect(panel3.api.location.type).toBe('grid');
dockview.addFloatingGroup(panel2);
expect(panel1.api.location.type).toBe('grid');
expect(panel2.api.location.type).toBe('floating');
expect(panel3.api.location.type).toBe('grid');
await dockview.addPopoutGroup(panel2);
expect(panel1.api.location.type).toBe('grid');
expect(panel2.api.location.type).toBe('popout');
expect(panel3.api.location.type).toBe('grid');
const windowObject =
panel2.api.location.type === 'popout'
? panel2.api.location.getWindow()
: undefined;
expect(windowObject).toBeTruthy();
windowObject!.close();
expect(panel1.api.location.type).toBe('grid');
expect(panel2.api.location.type).toBe('floating');
expect(panel3.api.location.type).toBe('grid');
});
test('move popout group of 1 panel inside grid', async () => { test('move popout group of 1 panel inside grid', async () => {
const container = document.createElement('div'); const container = document.createElement('div');
@ -5116,7 +5270,7 @@ describe('dockviewComponent', () => {
mockWindow.close(); mockWindow.close();
expect(panel1.group.api.location.type).toBe('grid'); expect(panel1.group.api.location.type).toBe('grid');
expect(panel2.group.api.location.type).toBe('grid'); expect(panel2.group.api.location.type).toBe('floating');
expect(panel3.group.api.location.type).toBe('grid'); expect(panel3.group.api.location.type).toBe('grid');
dockview.clear(); dockview.clear();

View File

@ -73,6 +73,7 @@ import {
OverlayRenderContainer, OverlayRenderContainer,
} from '../overlay/overlayRenderContainer'; } from '../overlay/overlayRenderContainer';
import { PopoutWindow } from '../popoutWindow'; import { PopoutWindow } from '../popoutWindow';
import { StrictEventsSequencing } from './strictEventsSequencing';
const DEFAULT_ROOT_OVERLAY_MODEL: DroptargetOverlayModel = { const DEFAULT_ROOT_OVERLAY_MODEL: DroptargetOverlayModel = {
activationSize: { type: 'pixels', value: 10 }, activationSize: { type: 'pixels', value: 10 },
@ -388,6 +389,10 @@ export class DockviewComponent
toggleClass(this.gridview.element, 'dv-dockview', true); toggleClass(this.gridview.element, 'dv-dockview', true);
toggleClass(this.element, 'dv-debug', !!options.debug); toggleClass(this.element, 'dv-debug', !!options.debug);
if (options.debug) {
this.addDisposables(new StrictEventsSequencing(this));
}
this.addDisposables( this.addDisposables(
this.overlayRenderContainer, this.overlayRenderContainer,
this._onWillDragPanel, this._onWillDragPanel,
@ -404,6 +409,7 @@ export class DockviewComponent
this._onDidRemoveGroup, this._onDidRemoveGroup,
this._onDidActiveGroupChange, this._onDidActiveGroupChange,
this._onUnhandledDragOverEvent, this._onUnhandledDragOverEvent,
this._onDidMaximizedGroupChange,
this.onDidViewVisibilityChangeMicroTaskQueue(() => { this.onDidViewVisibilityChangeMicroTaskQueue(() => {
this.updateWatermark(); this.updateWatermark();
}), }),
@ -571,6 +577,11 @@ export class DockviewComponent
this.updateWatermark(); this.updateWatermark();
} }
override dispose(): void {
this.clear(); // explicitly clear the layout before cleaning up
super.dispose();
}
override setVisible(panel: DockviewGroupPanel, visible: boolean): void { override setVisible(panel: DockviewGroupPanel, visible: boolean): void {
switch (panel.api.location.type) { switch (panel.api.location.type) {
case 'grid': case 'grid':
@ -706,6 +717,8 @@ export class DockviewComponent
_window.window!.innerHeight _window.window!.innerHeight
); );
let floatingBox: AnchoredBox | undefined;
if (!options?.overridePopoutGroup && isGroupAddedToDom) { if (!options?.overridePopoutGroup && isGroupAddedToDom) {
if (itemToPopout instanceof DockviewPanel) { if (itemToPopout instanceof DockviewPanel) {
this.movingLock(() => { this.movingLock(() => {
@ -727,7 +740,16 @@ export class DockviewComponent
break; break;
case 'floating': case 'floating':
case 'popout': case 'popout':
floatingBox = this._floatingGroups
.find(
(value) =>
value.group.api.id ===
itemToPopout.api.id
)
?.overlay.toJSON();
this.removeGroup(referenceGroup); this.removeGroup(referenceGroup);
break; break;
} }
} }
@ -825,20 +847,31 @@ export class DockviewComponent
}); });
} }
} else if (this.getPanel(group.id)) { } else if (this.getPanel(group.id)) {
this.doRemoveGroup(group, { const removedGroup = group;
if (floatingBox) {
this.addFloatingGroup(removedGroup, {
height: floatingBox.height,
width: floatingBox.width,
position: floatingBox,
});
} else {
this.doRemoveGroup(removedGroup, {
skipDispose: true, skipDispose: true,
skipActive: true, skipActive: true,
skipPopoutReturn: true, skipPopoutReturn: true,
}); });
const removedGroup = group;
removedGroup.model.renderContainer = removedGroup.model.renderContainer =
this.overlayRenderContainer; this.overlayRenderContainer;
removedGroup.model.location = { type: 'grid' }; removedGroup.model.location = { type: 'grid' };
returnedGroup = removedGroup; returnedGroup = removedGroup;
this.movingLock(() => {
// suppress group add events since the group already exists
this.doAddGroup(removedGroup, [0]); this.doAddGroup(removedGroup, [0]);
});
}
this.doSetGroupAndPanelActive(removedGroup); this.doSetGroupAndPanelActive(removedGroup);
} }
}) })
@ -1290,6 +1323,7 @@ export class DockviewComponent
locked: !!locked, locked: !!locked,
hideHeader: !!hideHeader, hideHeader: !!hideHeader,
}); });
this._onDidAddGroup.fire(group);
const createdPanels: IDockviewPanel[] = []; const createdPanels: IDockviewPanel[] = [];
@ -1306,8 +1340,6 @@ export class DockviewComponent
createdPanels.push(panel); createdPanels.push(panel);
} }
this._onDidAddGroup.fire(group);
for (let i = 0; i < views.length; i++) { for (let i = 0; i < views.length; i++) {
const panel = createdPanels[i]; const panel = createdPanels[i];
@ -1394,6 +1426,7 @@ export class DockviewComponent
'dockview: failed to deserialize layout. Reverting changes', 'dockview: failed to deserialize layout. Reverting changes',
err err
); );
/** /**
* Takes all the successfully created groups and remove all of their panels. * Takes all the successfully created groups and remove all of their panels.
*/ */

View File

@ -506,7 +506,9 @@ export class DockviewGroupPanelModel
this._onDidAddPanel, this._onDidAddPanel,
this._onDidRemovePanel, this._onDidRemovePanel,
this._onDidActivePanelChange, this._onDidActivePanelChange,
this._onUnhandledDragOverEvent this._onUnhandledDragOverEvent,
this._onDidPanelTitleChange,
this._onDidPanelParametersChange
); );
} }

View File

@ -0,0 +1,54 @@
import { CompositeDisposable } from '../lifecycle';
import { DockviewComponent } from './dockviewComponent';
export class StrictEventsSequencing extends CompositeDisposable {
constructor(private readonly accessor: DockviewComponent) {
super();
this.init();
}
private init(): void {
const panels = new Set<string>();
const groups = new Set<string>();
this.addDisposables(
this.accessor.onDidAddPanel((panel) => {
if (panels.has(panel.api.id)) {
throw new Error(
`dockview: Invalid event sequence. [onDidAddPanel] called for panel ${panel.api.id} but panel already exists`
);
} else {
panels.add(panel.api.id);
}
}),
this.accessor.onDidRemovePanel((panel) => {
if (!panels.has(panel.api.id)) {
throw new Error(
`dockview: Invalid event sequence. [onDidRemovePanel] called for panel ${panel.api.id} but panel does not exists`
);
} else {
panels.delete(panel.api.id);
}
}),
this.accessor.onDidAddGroup((group) => {
if (groups.has(group.api.id)) {
throw new Error(
`dockview: Invalid event sequence. [onDidAddGroup] called for group ${group.api.id} but group already exists`
);
} else {
groups.add(group.api.id);
}
}),
this.accessor.onDidRemoveGroup((group) => {
if (!groups.has(group.api.id)) {
throw new Error(
`dockview: Invalid event sequence. [onDidRemoveGroup] called for group ${group.api.id} but group does not exists`
);
} else {
groups.delete(group.api.id);
}
})
);
}
}

View File

@ -205,6 +205,8 @@ export abstract class BaseGrid<T extends IGridPanelView>
)(() => { )(() => {
this._bufferOnDidLayoutChange.fire(); this._bufferOnDidLayoutChange.fire();
}), }),
this._onDidMaximizedChange,
this._onDidViewVisibilityChangeMicroTaskQueue,
this._bufferOnDidLayoutChange this._bufferOnDidLayoutChange
); );
} }