-
Notifications
You must be signed in to change notification settings - Fork 145
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
Remove duplicate classes and excess whitespace #272
Conversation
Nice! Personally I would vote for enabling this by default and making it opt-out. Would be a breaking change but I think worth it because of how much this feature has been requested. |
We leave one space in when a class list consists of just whitespace
Yeah, I'm fine with enabling it by default. We'll want to release this as v0.6.0 if we do. |
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.
Looks good to me!
Only place where I can't make it work (which might be expected honestly) is in HTML files with handlebars-like interpolation.
For React, this seems to work as expected, input:
<div className={clsx('p-2 p-2', condition ? 'p-4 p-4' : '')} />
Output:
<div className={clsx('p-2', condition ? 'p-4' : '')} />
Same for Vue, works as expected, input:
<div :class="['p-2 p-2', condition ? 'p-4 p-4' : '']"></div>
Output:
<div :class="['p-2', condition ? 'p-4' : '']"></div>
However, if we do something similar in (post) HTML, then this doesn't seem to be the case.
Input:
<div class="p-2 p-2 {{ condition ? 'p-4 p-4' : '' }}"></div>
Actual:
<div class="p-2 p-2 {{ condition ? 'p-4 p-4' : '' }}"></div>
Expected:
<div class="p-2 {{ condition ? 'p-4' : '' }}"></div>
Not the end of the world, but if this is expected, then we might want to mention this in the description. It's also totally possible I'm missing something.
Yeah that's expected and not a side-effect of this PR. Prettier itself completely ignores class attributes with handlebars interpolation and we do the same. We could revisit that decision (in a separate PR) but we'd need to parse each attribute containing |
Any clue why these new features wouldn't work with .liquid files? |
@stijns96 It's due to the way the Liquid plugin works. Changing the length of the source string breaks re-printing of the AST in a number of places. Sometimes it uses information from the AST itself and sometimes it uses information from the original source text and a mismatch between them can cause problems. |
This PR introduces two new features to the plugin: removal of duplicate class names and removal of excess / unncecessary whitespace.
Removal of duplicate classes
This feature is always enabled. An example of this feature in action is shown below — notice that the extra
p-4
class is removed:Removal of excess / unnecessary whitespace
This was originally implemented in #70 but reverted due to issues breaking code that relies on the presence of whitespace in a class list. This PR enables this feature by default but allows you to disable it (via
tailwindPreserveWhitespace
) should you run into issues. For example, the following code will be formatted as shown below:What can break?
As mentioned above, removal of whitespace in a class list can break code that relies on the presence of leading or trailing whitespace. e.g. when concatenating strings of classes. In the following code, you can see the after version results in a non-existent class. This is a known limitation and is why this feature can be disabled.
To disable this feature and preserve whitespace in class lists, set the option
tailwindPreserveWhitespace
totrue
in your Prettier configuration:If you don't want to disable the feature for your entire project you can use one of the following methods:
// prettier-ignore
to prevent Prettier from formatting the line:When sorting we will ensure that a class list that is composed of only whitespace is not removed. It will, however, be shortened to a single space:
.prettierrc
file and it will only preserve whitespace insrc/components/MyComponent.jsx
: