-
Notifications
You must be signed in to change notification settings - Fork 4.3k
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
ToolsPanel: Allow SlotFill injection of panel items #34632
Conversation
Size Change: +284 B (0%) Total Size: 1.04 MB
ℹ️ View Unchanged
|
I see changes in the public API but it seems to be fine gutenberg/packages/components/src/index.js Lines 141 to 144 in e3591aa
|
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.
Thank you for moving changes to the ToolsPanel
to a separate PR, it made this review much easier!
Code changes in this PR LGTM :) Thank you also for adding more props to the docs, and for updating unit tests!
I was playing around with the component in Storybook, and probably came across a bug: deselecting a component doesn't seem to reset its value after clicking on the "Reset All" button:
toolspanel-reset.mp4
Also, do you think it'd be valuable to add a Storybook example showing how to use the panelId
props, especially in the context of injection of panel items via a SlotFill?
Although I'm ok with not adding it if you think it doesn't make much sense / bring much value.
Thanks for catching that @ciampo! When addressing the last round of feedback, I messed up the clearing of the I've added another unit test for this along with a fix. I now get the following behaviour: Screen.Recording.2021-09-09.at.2.34.42.pm.mp4
I've added another storybook example that creates its own SlotFill and uses that to illustrate injecting items into the panel. Hopefully, that makes sense and is useful. I'm not tied to it so if anyone has any suggestions on improving it or even removing it, that's fine by me. The rewording of the docs and some typo fixes have been made as well. |
|
||
return ( | ||
<SlotFillProvider> | ||
<ToolsPanelItems> |
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.
This example and the documentation efforts made me think whether panelId
should be part of the component. Should we create an opinionated Fill and Slot and keep them in the block editor instead?
The difference would be that ToolsPanelItem
from the @wordpress/block-editor
package would be a fill that handles all the logic related to panelId
and renders ToolsPanelItem
. There would be also a slot that would pass down panelId
to fills through fillProps
and would be wrapped with ToolPanel
component. I don't know if I'm doing a good job explaining it.
So the goal would be to achieve the following:
const { Fill, Slot } = createSlotFill( 'ToolsPanelSlot' );
function shouldRender( panelId ) {
// checks somehow whether to render the panel based on the value passed from slot using `fillProps`
}
const ToolsPanelItemFill = ( { panelId, ...props } ) => {
return (
<Fill>
{ shouldRender( panelId ) && <ToolsPanelItem { ...props } /> }
</Fill>
);
};
const ToolsPanelSlot = ( { panelId, ...props } ) => {
return (
<ToolsPanel { ...props }>
<Slot fillProps={ panelId } />
</ToolsPanel>
);
};
It needs more code, but I hope it helps to sketch the idea.
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.
That's an interesting idea. I don't think I fully understand the implications yet but will explore this tomorrow.
It makes sense in terms of being able to remove the need for panelId
on the ToolsPanel
itself. At this stage though, I don't have much of an idea how the individual ToolsPanelItem
s would be working.
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.
You probably can play with the Storybook story first and see if you can make it work this way. As noted in another place, given that both updated components are marked as experimental don't treat my comment as a blocker.
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.
Feel free to merge when @ciampo approves the recent changes 👍🏻
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.
Thank you @aaronrobertshaw for addressing all the feedback!
When addressing the last round of feedback, I messed up the clearing of the isResetting ref which ultimately caused the panel items to think the panel was still undertaking a "reset all" and therefore skip calling the onDeselect callbacks.
This explanation makes total sense and explains the bug — I can confirm that the component now behaves as expected!
I've added another storybook example that creates its own SlotFill and uses that to illustrate injecting items into the panel.
I agree with @gziolo's point about potentially extracting panelId
(and slot fill logic) to another part of the repo, but that's definitely not a blocker and should be investigated/tackled separately.
Also, special thanks for improving unit tests with every PR!
Co-authored-by: Marco Ciampini <[email protected]>
Related:
Description
This updates the
ToolsPanel
component to support injection of panel items via a SlotFill. The changes to theToolsPanel
here have been separated from #34157.Changes include:
ToolsPanelContext
so it can be passed via slots'fillProps
label
andheader
props for theToolsPanel
resetAll
callback allowing injected panel item controls to reset whatever they needHow has this been tested?
npm run test-unit packages/components/src/tools-panel/test
npm run storybook:dev
Screenshots
Types of changes
Enhancement. Bug fix.
Checklist:
*.native.js
files for terms that need renaming or removal).