-
Notifications
You must be signed in to change notification settings - Fork 14.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
[SIP-82] Improving Superset Theming #20159
Comments
Thanks for the comprehensive writeup. This is awesome! |
@michael-s-molina this is fantastic, so much great info and details in one big SIP ❤️ yay for Superset's UI consistency and flexibility! |
Great work!
Could test this much like the color tests, by changing all the theme's font sizes to zero. Anything visible is not using theme variables.
This option seems the best to me. We may also benefit from some new components Ant has added. Is runtime adjustable theming a planned feature for Ant 5? |
We're actually in a bad place right now with our current version of AntD (4.9.4):
|
Maybe this is too much to ask, but if upgrading AntD turned out to be too large an undertaking or if we have to write a lot more wrapper components, can we also consider migrating Superset to a more flexible and less opinionated UI library such as Chakra UI? I used Chakra in some other projects and was completely in love. Styled-component is a first-class citizen in Chakra. It already uses Emotion under the hood. Inline style props makes it very easy to override ad-hoc styles without bloating shared components. Customize theme with "variant" and "size" and multipart components provides a principled way of achieving visual consistency across components. Theming with Chakra really is a bliss and is light-year ahead of the current experience with AntD. Try it out yourself if you don't believe me! I know migrating to yet another UI library is a risky bet, but since we are on the verge of making a large investment in CSS/styling again, maybe it's worth at least doing some investigation on these other choices out there? |
I'm not a huge fan of big migrations, especially given the fact that we fairly recently migrated to AntD, but some thoughts on this:
|
This is a nice idea 👍🏼
Yes, it is.
Yes! We have some important upgrades to do with Ant Design and React. Especially because React 18 comes with important performance boosts.
I think it's a fair ask. As you said, it will depend on the amount of work necessary to achieve themeability. When making this decision, we'll have more information on Ant Design 5 as well, which will help with the evaluation. |
Great doc, really comprehensive and easy to read! |
Thanks for the context @kgabryje. The way I'm interpreting @kasiazjc's typography table is that previously defined "bold (600)" fonts will change to "semi-bold (600)" and "bold (700)" will only be used by "Monospace-strong" so we'll preserve the original look. |
I understand big migration is a huge risk but since we are talking about theming here, I'll still encourage you all to try out the theming approach in Chakra UI (which is inspired by Theme UI, btw), maybe even just for inspiration of how to apply themes to AntD. It'll blow your mind. |
Color and font is an intuitive change. However, most adjustments based on color and font are not ideal. Can we provide more advanced style changes? I know we should step by step. And I think the complete function of theme change (especially in China or else, which has some extreme demand for UI) is supported to such a level. |
This is really cool! Thanks for sharing this @cdmikechen. I totally understand the need to support advanced customizations. Think of this SIP as the first chapter of a book about Theming 😉. Before adding advanced features, we need to fix our foundation, but eventually, we'll get there! 🚀 |
This is major improvement!!! Any estimates when this will be ready? |
@catadedu the project represents a whole pile of "long tail" projects, so I think "done" is a bit of a way off. I'll try to detail the current state of affairs in a blog post since a lot of people have been asking some version of this question in various places. I think we're close to the React 18 prerequisite, which is one of many steps, for example. |
This is awesome @michael-s-molina. Somehow I missed this SIP when it first came out, and since then, but happy to see this and still think theming / "themability" is a super important topic. One question: say if I was to go all out on making Superset themeable, where would I start? Thinking about it, maybe I'd go full on brute-force using a dark-theme overriding Another approach would be to operate based on the design system we kind of have, and work on a component-by-component basis while using react-storybook. That way you can kind of break it down and into 1) making the design system fully themeable, and 2) which is a larger challenge -> making sure the design system is used everywhere in the app |
I'm still thinking about this after my previous comment! But an interesting question is - once we'd get to a fully themed/themable Superset, how do maintain this? How do we prevent regressions? This potentially adds a super complex dimension to the test matrix. It seems that making sure that all pieces of the app are fully themed and remain that way might be super challenging. |
There's actually a blog post on the topic too. There are a few "first things" to tackle, actually... I have a theory about how we can smoothly upgrade AntD, which is one of them... but that's dependant on rekindling visual regression testing... which is dependent on storybook... which is in the midst of a hairy upgrade. I have the feeling we'll find our way out of some of these logjams soon though. |
I would start with typography and making sure we can change it consistently throughout the application. This has the benefit of layering down the foundation for other steps, improves text standardization, and it's the shortest checkpoint that will already provide value.
I thought about the same strategy to validate the colors.
As @rusackas mentioned, visual regression testing might be the way to go here. |
@rusackas do you think that a key part of the strategy would be done through Maybe the way we fix higher level components that are NOT in the design system is by creating theme-compatible higher level components that are theme-compatible. |
Yes, absolutely, Storybook can be used exensively to facilitate the transition. First, it will allow us to make the upgrade to AntD more reliably (checking/testing individual component features, and using Applitools to check for visual regressions). Then, once AntD is upgraded and more theming-compatible, we can put test themes in Storybook (e.g. Dark Mode) and look for problems with each and every core component. |
screencap here walking through the work so far -> https://drive.google.com/file/d/1KKQf6qH70dKhSmZEMBAL5TEK71vA7H_k/view?usp=sharing |
@mistercrunch I'm fully supportive of moving towards vanilla Ant Design components. I completely agree that our customizations have a high maintenance cost. @geido and I discussed this problem many times and we reached the same conclusion. I believe it's important to get @kasiazjc's approval as well given that we'll have more limitations in terms of design by following vanilla Ant Design (which I think it's a good thing). |
I'll keep pushing in this direction in #31590, which is pretty major step in that direction and viable pretty much as-is (I just need to fix the cypress tests I've broken along the way 😰). I'll add more notes in that PR, but I added a dark/light theme switcher in the navbar (behind a feature flag) in there which makes it easy clear what's dynamic css-in-js and what isn't. It feels great to rip all that less and css-in-js and go vanilla. I'd say the goal is for a Superset theme object to effectively be an antd ThemeConfig, augmented with the few things that are important to Superset's theming, yet out-of-scope for something as generic as AntD. Things like |
One important thing to remember is we will also need to provide a theme object for ECharts. So if we're confident the AntD |
Totally. From what I see it seems very doable to extend the antd theme system with custom tokens and custom components. From what I see you can totally write a ThemeConfig like this: const themeConfig = {
token: {
primaryColor: 'red', // this is a base token they use internally in their components
textColorBase: '#AAA', // another seed token I think
customGlobalToken: 'blue', // this item isn't in the official list
hackyNestedObject: {'hello': 'world'}, // this seems to also works, but diverges from their flat approach to tokens. potentially could be used to squish a whole echartOptions-compatible nested structure...
}
components: {
Button: { // an official component
primaryColor: 'brown', // this is the prescribed way to override a token for a specific component
},
ECharts: { // this component isn't a registered antd component but will be made available
echartSpecificTextSize: '20px',
textColorBase: '#EEE', //override of textColorBase specific to echarts
}
}
} all of which become available through antd's So in any component, you can just: // retrieving custom token in component module
const { customGlobalToken } = theme.useToken();
// retrieving Echarts specific stuff
const { textColorBase, echartSpecificTextSize } = theme.useToken().componentss.Echarts; I couldn't find a function that merges/flattens tokens in the context of a specific component say |
Started this discussion on antd's repo -> ant-design/ant-design#52270 |
[SIP-82] Improving Superset Theming
Motivation
Recently we did a POC with the objective of changing the colors of the Superset theme to analyze how it affected the application. The reason was to understand how far we are from allowing Superset users to customize the theme. The objective of this SIP is to document the findings of the POC and to propose some efforts to improve Superset theming capabilities. We understand that some of the efforts may need additional SIPs or are linked with existing SIPs. It’s not the objective of this SIP to dive into the details of all the efforts but to propose an overview of the work necessary and to propose solutions to some of them.
The Superset theme configuration has the following structure:
We started by changing the theme's
primary
,secondary
, andgrayscale
colors to green, purple, and gold respectively using colors from Ant Design color palettes.We also changed text colors to magenta. Here's the result of these changes on some of our main screens:
Some interesting conclusions can be drawn from the above images:
Proposed Change
To deal with these problems and make Superset themeable, we need to complete some efforts.
Standardize theme access
If we do a quick search for “supersetTheme" we can see that the theme object is accessed directly in some files. This means that these files will not be affected by potential changes to theming properties. At the time of writing, we have 40 files referencing the theme directly. We need to fix these files and standardize theme access.
Get rid of the remaining LESS files
We need to complete our work and remove the remaining LESS files from the application. At the time of writing, we have 17 LESS files remaining. Given that each LESS file contains many style definitions that may or may not be in use by the application, this task will require a great deal of effort and can potentially introduce regressions.
Get rid of the remaining CSS files
Some plugins are using CSS files to configure the theme properties. We need to migrate them to Emotion and make the necessary adjustments to colors, fonts, etc. At the time of writing, we have about 20 CSS files.
Remove hard-coded typography and colors
Many components are still using hard-coded typography (font families, weights, sizes) and colors. At the time of writing, we don't have an estimate on how many components need to be fixed.
Write linting rules to find and warn on non-themed styles we care about in Emotion
One effort that can help us find non-themed styles and prevent future incompatibilities is to write linting rules that are capable of finding these errors. It is worth spending some time trying to make this work because it can really buy us time in the short and long term.
Change the way we import Ant Design from LESS to CSS
Right now we are importing Ant Design LESS files directly to override some of its variables. As we progress with removing LESS dependencies we also need to change the way we import Ant Design to use the CSS version as described in the installation docs.
Migrate server-side pages to the client-side
Many Superset pages are still being rendered on the server-side. Some examples include:
Right now these pages aren't affected by theming changes. We considered maintaining these pages on the server-side and try to apply the theme configurations to them. After analyzing the complexities and limitations of the solution, we recommend migrating them to the client-side since this is also a requirement for the single-page application (SPA) effort. As part of this work, we may need new APIs to support/hydrate these pages.
Improve the theme structure definition
The theme structure definition constitutes an API between Superset and theme providers. A poorly defined structure will make it difficult for theme providers to understand which parts of the application are affected by the changes, may limit the areas for customization, and can lead to unnecessary migration while evolving.
Looking at the structure of our current theme, we found that we needed to make some improvements to support multiple themes. To come up with the improvement suggestions we took inspiration from the following references during the research phase:
To support multiple themes, we are suggesting the following changes to the
colors
theme structure:Create a
background
category with thedefault
,surface
, andpopover
subcategoriesWe noticed that we need these categories to support the customization of background structures like dashboard cards, popovers, tooltips, navigation panels, etc. These colors are independent of the primary and grayscale categories. This became really clear when simulating a dark theme.
Modify the
text
categoryWe need to add more text categories to reflect the different use cases found in the application. The
default
category is used to display the majority of texts in the application. We also created thehint
anddisabled
categories and removed thelabel
andhelp
categories.To deal with texts that appear on top of primary and grayscale colors we'll use algorithms to automatically determine the most appropriate color.
Remove the secondary color
Currently, the secondary color is rarely seen in Superset. We're proposing to adjust these rare places to use primary and grayscale colors and remove the secondary category, simplifying the theme configuration.
Merge the
warning
andalert
categoriesWe can simplify the configuration of our functional colors by merging the warning and alert categories. The idea is to keep the
warning
category with the colors from thealert
category because both the name and the gold colors are most commonly found in other applications and align better with the users’ basic understanding of color functions.We can also improve typography definitions. Although we have families, weights, and sizes, it's not always clear where they are being used and how they are being compounded to create common text elements. Because the composition is not clearly established, developers tend to mix different weights and sizes for components that should belong to the same typography category.
To improve the standardization of typography categories and at the same time maintain the flexibility of the atomic configurations, we propose to enhance our theme with the introduction of
presets
. Presets are a combination of families, weights, sizes, and line heights that form the typography building blocks of the application. They are not arbitrarily defined but follow the design system specification which means that any change to them should be a reflection of a change in the design system. One important step we can take to make sure this statement remains true in the future is to explicitly set code owners for the theme files with members of the design team and selected engineers.Here’s the typography building blocks as defined by the design system (@kasiazjc):
The proposed changes are to adjust the
weights
andsizes
categories to reflect the design system and to create the presets category:With these definitions, we can easily map them to the screens and maintain a consistent design throughout the application:
The suggested changes here constitute the first draft for the theme categories and are not complete. As this is an iterative approach, we’re open to further modifications to the theme structure while working on theming tasks. We can use the dark theme implementation as the compass to define if we need additional categories or if further simplifications can be made.
Define a way to apply our theme to Ant Design's theme
One problem we currently have is how to customize Ant Design's theme directly. Right now we have many components that override the native styles by changing Ant Design CSS properties manually. Some components are extensions to the native components and add new features or modify the way a component behaves. Others are simply an extra layer to change CSS. The problem with that approach is that it increases the probability of breaking changes when upgrading Ant Design versions. One of the bigger problems right now is that Ant Design uses LESS for defining theme variables and the support for dynamically changing these variables is really poor. Luckily, after many requests from the community, the Ant Design team finally decided to support CSS-in-JS in Ant Design 5. This will probably change the way we are modifying native styles and reduce the number of custom components on our end. We don't have an estimate for this version yet so we need to come up with a temporary solution. During the POC we come up with three different possibilities:
Upgrade Ant Design to the latest version and use CSS variables
One possibility is to use the new CSS variables feature available in the latest version of Ant Design. This feature allows us to dynamically override theme properties using a
ConfigProvider
. The downside of this feature is that only some properties are actually customizable, which means this solution may not be sufficient for our use case. We would also need to upgrade Ant Design to the latest version and that would require adjustments to our application to make sure the styles are still working. The advantage of this approach is that we won't expand our layer of custom components, preventing work that may be discarded when upgrading to Ant Design 5. Also, it would be a catalyst to finally get visual testing integrated into our development environment. Migrating to the latest version would also help prepare the code base for the next major version upgrade.Complete the layer of custom components
Another possibility is complete our layer of custom components by adding mappings to Ant Design components that are currently being accessed directly. This has the disadvantage of expanding our custom layer even more which may change after upgrading to Ant Design 5. The advantages are that we don't need to upgrade Ant Design, avoiding all the migration errors, and the level of possible customization.
Wait for Ant Design version 5
We can also wait for the next major version and limit the scope of what we can customize. During the POC it was hard to determine if the areas that were not affected by theming changes are the result of code that is not using the theme correctly or if it's related to Ant Design styles.
Make use of global styles to override specific components
We could also use global styles to fill in the blanks and override specific components while keeping the Ant Design version. This could be a temporary patch until we wait for version 5.
Conclusion
I think the decision of which approach to follow should be made after we fix incorrect code and check the remaining non-themeable parts of the application. Then we can fully determine which approach is faster, given that the new Ant Design version will probably change the solution.
Having visual regression testing enabled for this work would be essential to mitigate potential regressions. We should make it a pre-requirement for many of the efforts described here.
Apply our theme to ECharts plugins
ECharts supports theme customization by providing and registering a theme object or using a predefined theme and explicitly setting style options.
Here's a preview of the ECharts theme structure obtained by downloading the dark theme:
In our case, we recommend using the default theme and mapping our theme configurations to each category defined by ECharts. Currently, each plugin is passing these ECharts configurations via an
options
object that is calculated by thetransformProps
andloadTransformProps
functions. Right now, each plugin is creating its own version of theoptions
object. To allow customization of properties that are applied to all plugins we can initialize theEchart
component with a baseoptions
object and complement its properties with the ones coming from the plugins. We also need to remove base configurations from the plugins unless we're specifically overriding the behavior. Here's an example:As you can see from the examples, we need to make the Superset theme available for both functions.
Apply our theme to non-ECharts plugins
To apply our theme to plugins that are not using the ECharts library, we'll need to evaluate each plugin individually and determine what's the best approach, whether is to manually apply CSS classes to the relevant parts of the plugin (e.g. Pivot Table) or use the plugins' API to change theming properties (e.g. Word Cloud).
Create a dark theme to test the application
Creating a dark theme will be a good test for Superset themeability. Using a theme with opposite colors enables us to validate if all UI elements are being affected by theming changes. This is especially useful to check text, borders, shadows, and background colors with different contrasts. We could also make the dark theme available as one of the default themes after the work is completed.
Create a wiki page on how to customize Superset
Organizing the theme structure helps a lot but there's nothing better than a proper theming guide for Superset users. We're suggesting a wiki page with all the information about theme categories, which parts of the application are affected by each category, instructions on how to manage themes, etc.
Future work
In the future, we could extend the work to allow administrators to manage multiple themes via a specific module in Superset. We could have features similar to ECharts theme builder or Material UI theme creator for editing theme configurations. We could also allow administrators to export/import themes to promote theme sharing.
New or Changed Public Interfaces
New dependencies
Depending on the output of our research on how to apply our theme to Ant Design, we may have to upgrade its version.
Migration Plan and Compatibility
No database migrations or updates to stored URLs are necessary.
Rejected Alternatives
Special Thanks
Special thanks to @kasiazjc @geido @rusackas @mihir174 @jess-dillard @villebro and all the other reviewers for helping with the work.
The text was updated successfully, but these errors were encountered: