Skip to content

Commit bcdb2bc

Browse files
authored
[core] feat(PanelStack): add renderCurrentPanelOnly prop (#3768)
1 parent f680c5d commit bcdb2bc

File tree

4 files changed

+59
-9
lines changed

4 files changed

+59
-9
lines changed

packages/core/src/components/panel-stack/_panel-stack.scss

+5-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@
1313
display: flex;
1414
flex-shrink: 0;
1515
align-items: center;
16-
z-index: 1;
1716
box-shadow: 0 1px $pt-divider-black;
1817
height: $pt-grid-size * 3;
1918

@@ -48,6 +47,7 @@
4847
@include position-all(absolute, 0);
4948
display: flex;
5049
flex-direction: column;
50+
z-index: 1;
5151

5252
// border between panels, visible during transition
5353
margin-right: -1px;
@@ -59,6 +59,10 @@
5959
.#{$ns}-dark & {
6060
background-color: $dark-gray4;
6161
}
62+
63+
&:nth-last-child(n + 4) {
64+
display: none;
65+
}
6266
}
6367

6468
// PUSH transition: enter from right (100%), existing panel moves off left.

packages/core/src/components/panel-stack/panelStack.tsx

+24-8
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,12 @@ export interface IPanelStackProps extends IProps {
4646
*/
4747
onOpen?: (addedPanel: IPanel) => void;
4848

49+
/**
50+
* If false, PanelStack will render all panels in the stack to the DOM, allowing their React component trees to maintain state as a user navigates through the stack. Panels other than the currently active one will be invisible.
51+
* @default true
52+
*/
53+
renderCurrentPanelOnly?: boolean;
54+
4955
/**
5056
* Whether to show the header with the "back" button in each panel.
5157
* @default true
@@ -98,7 +104,7 @@ export class PanelStack extends AbstractPureComponent2<IPanelStackProps, IPanelS
98104
);
99105
return (
100106
<TransitionGroup className={classes} component="div">
101-
{this.renderCurrentPanel()}
107+
{this.renderPanels()}
102108
</TransitionGroup>
103109
);
104110
}
@@ -115,25 +121,35 @@ export class PanelStack extends AbstractPureComponent2<IPanelStackProps, IPanelS
115121
}
116122
}
117123

118-
private renderCurrentPanel() {
119-
const { showPanelHeader = true } = this.props;
124+
private renderPanels() {
125+
const { renderCurrentPanelOnly = true } = this.props;
120126
const { stack } = this.state;
121127
if (stack.length === 0) {
122128
return null;
123129
}
124-
const [activePanel, previousPanel] = stack;
130+
const panelsToRender = renderCurrentPanelOnly ? [stack[0]] : stack;
131+
const panelViews = panelsToRender.map(this.renderPanel).reverse();
132+
return panelViews;
133+
}
134+
135+
private renderPanel = (panel: IPanel, index: number) => {
136+
const { showPanelHeader = true } = this.props;
137+
const { stack } = this.state;
138+
const active = index === 0;
139+
const layer = stack.length - index;
140+
const key = `${layer}-${active}`;
125141
return (
126-
<CSSTransition classNames={Classes.PANEL_STACK} key={stack.length} timeout={400}>
142+
<CSSTransition classNames={Classes.PANEL_STACK} key={key} timeout={400}>
127143
<PanelView
128144
onClose={this.handlePanelClose}
129145
onOpen={this.handlePanelOpen}
130-
panel={activePanel}
131-
previousPanel={previousPanel}
146+
panel={panel}
147+
previousPanel={stack[index + 1]}
132148
showHeader={showPanelHeader}
133149
/>
134150
</CSSTransition>
135151
);
136-
}
152+
};
137153

138154
private handlePanelClose = (panel: IPanel) => {
139155
const { stack } = this.state;

packages/core/test/panel-stack/panelStackTests.tsx

+21
Original file line numberDiff line numberDiff line change
@@ -234,6 +234,27 @@ describe("<PanelStack>", () => {
234234
assert.equal(firstPanelHeader.at(0).text(), "Test Title");
235235
});
236236

237+
it("renders only one panel by default", () => {
238+
const stack = [{ component: TestPanel, title: "Panel A" }, { component: TestPanel, title: "Panel B" }];
239+
panelStackWrapper = renderPanelStack({ stack });
240+
241+
const panelHeaders = panelStackWrapper.findClass(Classes.HEADING);
242+
assert.exists(panelHeaders);
243+
assert.equal(panelHeaders.length, 1);
244+
assert.equal(panelHeaders.at(0).text(), stack[1].title);
245+
});
246+
247+
it("renders all panels with renderCurrentPanelOnly disabled", () => {
248+
const stack = [{ component: TestPanel, title: "Panel A" }, { component: TestPanel, title: "Panel B" }];
249+
panelStackWrapper = renderPanelStack({ renderCurrentPanelOnly: false, stack });
250+
251+
const panelHeaders = panelStackWrapper.findClass(Classes.HEADING);
252+
assert.exists(panelHeaders);
253+
assert.equal(panelHeaders.length, 2);
254+
assert.equal(panelHeaders.at(0).text(), stack[0].title);
255+
assert.equal(panelHeaders.at(1).text(), stack[1].title);
256+
});
257+
237258
interface IPanelStackWrapper extends ReactWrapper<IPanelStackProps, any> {
238259
findClass(className: string): ReactWrapper<React.HTMLAttributes<HTMLElement>, any>;
239260
}

packages/docs-app/src/examples/core-examples/panelStackExample.tsx

+9
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ import { Button, H5, Intent, IPanel, IPanelProps, PanelStack, Switch, UL } from
2020
import { Example, handleBooleanChange, IExampleProps } from "@blueprintjs/docs-theme";
2121

2222
export interface IPanelStackExampleState {
23+
activePanelOnly: boolean;
2324
currentPanelStack: IPanel[];
2425
showHeader: boolean;
2526
}
@@ -34,15 +35,22 @@ export class PanelStackExample extends React.PureComponent<IExampleProps, IPanel
3435
};
3536

3637
public state = {
38+
activePanelOnly: true,
3739
currentPanelStack: [this.initialPanel],
3840
showHeader: true,
3941
};
4042

43+
private toggleActiveOnly = handleBooleanChange((activePanelOnly: boolean) => this.setState({ activePanelOnly }));
4144
private handleHeaderChange = handleBooleanChange((showHeader: boolean) => this.setState({ showHeader }));
4245

4346
public render() {
4447
const stackList = (
4548
<>
49+
<Switch
50+
checked={this.state.activePanelOnly}
51+
label="Render active panel only"
52+
onChange={this.toggleActiveOnly}
53+
/>
4654
<Switch checked={this.state.showHeader} label="Show panel header" onChange={this.handleHeaderChange} />
4755
<H5>Current stack</H5>
4856
<UL>
@@ -59,6 +67,7 @@ export class PanelStackExample extends React.PureComponent<IExampleProps, IPanel
5967
initialPanel={this.state.currentPanelStack[0]}
6068
onOpen={this.addToPanelStack}
6169
onClose={this.removeFromPanelStack}
70+
renderCurrentPanelOnly={this.state.activePanelOnly}
6271
showPanelHeader={this.state.showHeader}
6372
/>
6473
</Example>

0 commit comments

Comments
 (0)