-
-
Notifications
You must be signed in to change notification settings - Fork 47
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
[react] Create Grid component #144
Conversation
@@ -0,0 +1,43 @@ | |||
'use client'; |
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 is the only demo from the docs demos that is a client component, all others run on the next app as RSC 🚀
The reason this has to be a client component is that the demo uses state to interactively show spacing 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.
AWESOME!!! ❤️
'--Grid-self-column-spacing': ['--Grid-self-column-spacing'], | ||
'--Grid-self-row-spacing': ['--Grid-self-row-spacing'], | ||
'--Grid-self-offset': ['--Grid-self-offset'], | ||
'--Grid-self-margin-left': ['--Grid-self-margin-left'], |
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.
Yes, this is a lot of variables 😅
But this is the only way I see we can support all the props being breakpoint aware: columns, spacing, size, offset, as well as nesting grids.
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.
Not a problem IMO 👍
conditions: Object.keys(theme.breakpoints.values).reduce((acc, breakpoint) => { | ||
acc[breakpoint] = `@media (min-width: ${theme.breakpoints.values[breakpoint]}${ | ||
theme.breakpoints.unit ?? 'px' | ||
})`; | ||
return acc; | ||
}, {}), | ||
defaultCondition: theme.breakpoints?.keys?.[0] ?? 'xs', | ||
properties: { | ||
flexDirection: ['column', 'column-reverse', 'row', 'row-reverse'], |
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 is the same as the Stack but I'm not sure we should abstract it, it might be too early. A little duplication is not that bad in my opinion.
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.
I agree with no abstraction. Each component should have independent config.
@@ -62,7 +65,7 @@ describe('generateAtomics', () => { | |||
).to.deep.equal({ | |||
className: 'gap--Stack-gap-lg gap--Stack-gap-xs', | |||
style: { | |||
'--Stack-gap': 'calc(2 * 8px)', | |||
'--Stack-gap-xs': 'calc(2 * 8px)', |
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.
I had to bring this back because there are some variable to variable mappings, for example --Grid-self-width
that can be assigned to --Grid-self-width-xs
, so we need different names from the variable that points to the breakpoint variables.
As explained in the description, the APIs from Grid v2 to the Pigment Grid differ. It's easy to cover these differences with a codemod, which I think we should provide. The question would be should we implement this new API in the Grid v2? |
Please see my suggestion, I'd prefer to keep the existing APIs as much as possible. I imagine that developers might switch back and forth between Emotion/Pigment CSS in v6. |
GridAtomicsObj['--Grid-self-column-span'] = size; | ||
GridAtomicsObj['--Grid-self-width'] = size; | ||
GridAtomicsObj['--Grid-self-max-width'] = size; | ||
GridAtomicsObj['--Grid-self-flex'] = size; |
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.
I think it is possible to support the existing xs
, sm
, … xl
props without changing much code.
GridAtomicsObj['--Grid-self-column-span'] = size; | |
GridAtomicsObj['--Grid-self-width'] = size; | |
GridAtomicsObj['--Grid-self-max-width'] = size; | |
GridAtomicsObj['--Grid-self-flex'] = size; | |
GridAtomicsObj['--Grid-self-column-span'] = props; | |
GridAtomicsObj['--Grid-self-width'] = props; | |
GridAtomicsObj['--Grid-self-max-width'] = props; | |
GridAtomicsObj['--Grid-self-flex'] = props; |
GridAtomicsObj['--Grid-self-offset'] = offset; | ||
GridAtomicsObj['--Grid-self-margin-left'] = offset; |
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.
GridAtomicsObj['--Grid-self-offset'] = offset; | |
GridAtomicsObj['--Grid-self-margin-left'] = offset; | |
const offsetProps = {}; | |
Object.keys(props).forEach(prop => { | |
if (prop.endsWith('Offset')) { | |
offsetProps[prop.replace('Offset', '')] = props[prop] | |
} | |
}) | |
GridAtomicsObj['--Grid-self-offset'] = offsetProps; | |
GridAtomicsObj['--Grid-self-margin-left'] = offsetProps; |
Please add the test for the component. You can follow this pattern. |
@DiegoAndai about the API, some thoughts and questions:
|
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.
Left 2 minor nits. Rest all looks good. Nice work on implementing Grid. Next effort should be to have the similar API on runtime Grid as well for all layout components.
@siriwatknp yes! your suggestion would work. I thought of implementing it that way, but I agree with @aarongarciah:
In other words, I think having So while I agree that the APIs should be the same if possible, I suggest keeping this API, deprecating (or removing) the breakpoint props in Grid v2 ( On a side note, we could also stabilize the Grid v2 and maybe deprecate the Grid v1? What do you think? Do you see any reason not to follow this approach? |
To me, if the change in the API fixes issues I would consider going for it. However, it sounds like the change is based on our assumption that it is better. So if we compare "better API" with "more work + breaking change", I'd go with keeping the same API for now. To the people who want to upgrade to v6, these breaking changes might not be better for them (since it is not fixing any issue). Even though we have codemod, it could not be enough. You will be surprised how fancy the codebase could look like when they open an issue that the codemod does not work 😅. What do you think @mnajdova? Have you seen any issues regarding the existing API since older version @oliviertassinari? |
I wouldn't say it's an assumption. It's objectively more consistent with the
This is a good moment to improve the API: we are working on a new major, introducing an alternative no-runtime component, and Grid v2 is currently marked as unstable. This is the moment when we're focused on the layout components, I don't know if there will be another major soon that is focused on them. |
Agree with Diego here. We can pitch this as a upcoming stable equivalent of Grid v2 and we have this time to do the API changes that we want. |
Hey @siriwatknp! I added the Grid tests setup. I wanted to add more tests to // fixtures/Grid.output.js
import _default from '@pigment-css/react';
// ...
const GridComponent = /*#__PURE__*/ _default('div')({
// ... Which fails with:
I noticed the Container fixture also has this issue, so importing it doesn't work either. Do you know what might be going on? cc: @brijeshb42 |
@DiegoAndai Modifying line 419 in const styledImportIdentifier = t.addNamedImport('styled', process.env.PACKAGE_NAME as string); should fix your issue. |
Thanks @brijeshb42, it works now 👌🏼 |
This is ready, the only thing left to decide is the API discussion. |
To adopt the new API
Let's vote, 👍 for Yes and 😕 for No. cc @DiegoAndai @aarongarciah @brijeshb42 @mnajdova (@danilo-leal @zanivan tag both of you for user perspective) |
I want to justify my vote, as I haven't leave any comment so far. I am voting for the new API as it feels better, it's closer to the sx prop syntax that people are already familiar with. This is a major version, and the change is somewhat related to providing a support for new styling engine, considering we don't have almost any breaking change, it makes sense to include it, rather then adding more technical dept if we do want to change the API anyway in the next major. One more reason is that we are talking about the |
@DiegoAndai based on #144 (comment), let's move forward with the new API. Can you update the Unstable_Grid in a separate PR? |
Yes, I'll merge this today and open the PR to update the Unstable_Grid |
@DiegoAndai I think |
Created PR #159 for adding a wrap prop |
Create Grid component following the atomics approach, same as #118.
Summary
This is the Pigment version of the Grid v2 component. It supports all the use cases the original supports but has some API differences:
Sizing
Instead of using the breakpoints as props for size (
xs
,sm
,md
, ...), there's asize
prop. Some conversionsxs={2}
turns intosize={2}
xs={2} md={4}
turns intosize={{ xs: 2, md: 4 }}
xs
turns intosize='grow'
xs='auto'
turns intosize='auto'
xs={2} sm md={4} lg='auto'
turns intosize={{ xs: 2, sm: 'grow', md: 4, lg: 'auto' }}
Offset
Instead of using the breakpoints as props for offset (
xsOffset
,smOffset
,mdOffset
, ...), there's aoffset
prop. Some conversionsxsOffset={2}
turns intooffset={2}
xsOffset={2} mdOffset={4}
turns intooffset={{ xs: 2, md: 4 }}
xsOffset='auto'
turns intooffset='auto'
xsOffset={2} mdOffset={4} lgOffset='auto'
turns intooffset={{ xs: 2, md: 4, lg: 'auto' }}
This was done as it works better with the no runtime approach. It might be possible to maintain the previous API, but I don't think it's worth it as it might introduce some tech debt, and this API is more consistent IMO. It is also easy to cover with a codemod.
Unstable internal props.
The original Grid v2 uses a
unstable_level
prop for nesting grids. This uses a similar approach but instead of levels it takes advantage of CSS variable scoping. This is the only way I could find to achieve nested grids. The problem it solves is that when a Grid is both a container and an item, it has to read from and set the column count.There might be alternatives to remove the use of these unstable internal props and check the component
muiName
, but I couldn't find a better way and I think we should move forward now. If someone has an idea for alternative implementations, I will test them. I don't think using:has
would help.Play with the Grid
The demo is at
http://localhost:5173/grid
and the code is atapps/pigment-css-vite-app/src/pages/grid.tsx
Preview
I've copied all demos from the docs to the test apps.
tinywow_tinywow_Screen.Recording.2024-06-13.at.17.27.12_58198239_58198336.mov