-
Notifications
You must be signed in to change notification settings - Fork 915
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
Feat (core): Make theme settings user-specific and user-configurable #5652
Changes from 26 commits
e3d494f
b138518
b0af131
90fa4a6
5586192
da426c2
f073016
f95c482
db5b415
681afca
eb5b2ae
a824708
ee7a04c
3290e4a
0f57b88
0586ae0
d9dc09a
f49765d
1782c6a
077424c
21a8740
c1987a9
c424a21
7377ca9
8283561
f56628a
abd6fb9
f0032cf
bea86f8
97876e3
ac16643
1510505
7b71d55
ca2a00e
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,131 @@ | ||
# Theme System | ||
|
||
## Basic concepts | ||
|
||
### Theme definitions in OUI | ||
|
||
Themes are defined in OUI via https://github.com/opensearch-project/oui/blob/main/src/themes/themes.ts. When Building OUI, there are several theming artifacts generated (beyond the react components) for each mode (light/dark) of each theme: | ||
|
||
1. Theme compiled stylesheets (e.g. `@elastic/eui/dist/eui_theme_dark.css`). Consumed as entry files in [/packages/osd-ui-shared-deps/webpack.config.js](/packages/osd-ui-shared-deps/webpack.config.js) and republished by `osd-ui-shared-deps` (e.g. [UiSharedDeps.darkCssDistFilename](/packages/osd-ui-shared-deps/index.js)). | ||
2. Theme compiled and minified stylesheets (e.g. `@elastic/eui/dist/eui_theme_dark.min.css`). These appear unused by OpenSearch Dashboards | ||
3. Theme computed SASS variables as JSON (e.g. `@elastic/eui/dist/eui_theme_dark.json`). Consumed by [/packages/osd-ui-shared-deps/theme.ts](/packages/osd-ui-shared-deps/theme.ts) and made available to other components via the mode and theme aware `euiThemeVars`. In general, these should not be consumed by any other component directly. | ||
4. Theme type definition file for SASS variables as JSON (e.g. `@elastic/eui/dist/eui_theme_dark.json.d.ts`) | ||
|
||
Note that all of these artifacts should ideally only be imported or used directly in one place (by `osd-ui-shared-deps`). | ||
|
||
In addition to these artifacts, OpenSearch Dashboards also makes heavy use of the theme SASS variables and mixins as defined in the source files (e.g. `@elastic/eui/src/theme_dark.scss`). | ||
|
||
### Theme definitions in OpenSearch Dashboards | ||
|
||
1. Theme tags are defined in [/packages/osd-optimizer/src/common/theme_tags.ts](/packages/osd-optimizer/src/common/theme_tags.ts) corresponding to each mode (light/dark) of each OUI theme. | ||
AMoo-Miki marked this conversation as resolved.
Show resolved
Hide resolved
|
||
2. These tags must correspond to entrypoint SCSS files in [/src/core/public/core_app/styles/](/src/core/public/core_app/styles/_globals_v8dark.scss), because they are imported by all SCSS files as part of the `sass-loader` in [/packages/osd-optimizer/src/worker/webpack.config.ts](/packages/osd-optimizer/src/worker/webpack.config.ts) and [/packages/osd-optimizer/src/worker/theme_loader.ts](/packages/osd-optimizer/src/worker/theme_loader.ts). Note that the optimizer webpack will compile a separate stylesheet for each unique mode and theme combination. | ||
3. OUI SCSS source files are also imported by `osd-ui-framework`, which generates the legacy KUI stylesheets (e.g. [/packages/osd-ui-framework/src/kui_next_dark.scss](/packages/osd-ui-framework/src/kui_next_dark.scss)). KUI is a UI library that predates EUI/OUI, and should be deprecated and fully removed via [#1060](https://github.com/opensearch-project/OpenSearch-Dashboards/issues/1060). Because it's a legacy package it has its own build process that doesn't use webpack; it just [compiles the SCSS files with grunt](/packages/osd-ui-framework/Gruntfile.js). But similarly to 2., a separate stylesheet is generated for each mode and theme combination. | ||
|
||
### Thmemed assets in OpenSearch Dasboards | ||
|
||
In general, most themed assests can be found in [/src/core/server/core_app/assets](src/core/server/core_app/assets/fonts/readme.md) (it also includes non-themed assets such as `favicons`, which could easily be themed if desired in the future). | ||
|
||
Most of the graphics/images are only dark/light mode-specific, not theme-specific: | ||
|
||
1. `default_branding` marks | ||
2. `logos` | ||
|
||
This directory also includes legacy CSS files ([/src/core/server/core_app/assets/legacy_dark_theme.css](/src/core/server/core_app/assets/legacy_dark_theme.css) and [/src/core/server/core_app/assets/legacy_light_theme.css](/src/core/server/core_app/assets/legacy_light_theme.css)), which predate even KUI, and are still used by some plugins (notably `discover`). See [#4385](https://github.com/opensearch-project/OpenSearch-Dashboards/pull/4385) for an experiment in removing these. Unlike KUI, they don't rely on OUI themes at all. | ||
|
||
Finally, font assets are a bit of a special case. Theme-specific fonts are defined by OUI, but it doesn't include the font definitions directly. Instead, the font assets are in [/src/core/server/core_app/assets/fonts](/src/core/server/core_app/assets/fonts/readme.md). The corresponding `@font-face` style definitions are generated at runtime via [/src/core/server/rendering/views/fonts.tsx](/src/core/server/rendering/views/fonts.tsx). | ||
|
||
## Theme settings | ||
|
||
## Theme loading | ||
|
||
```mermaid | ||
sequenceDiagram | ||
autonumber | ||
critical Setup | ||
core/server->>core/server/rendering: setup rendering service | ||
core/server/rendering->>core/server: provide render() method | ||
core/server->>core/server: setup legacy service | ||
core/server->>legacy: create legacy server | ||
legacy->>legacy: start ui mixin to<br>handle special routes | ||
core/server->>core/server/core_app: setup core app | ||
core/server/core_app->>core/server/core_app: register default routes | ||
core/server/core_app->>core/server/core_app: register static asset dir routes | ||
end | ||
Browser->>core/server: OSD page request (e.g. /app/home#/ ) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. wow - well spotted 👁️! I'll update on the next revision. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @joshuarrrr would you have time to address this? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @AMoo-Miki i have created an issue for non-blockers to which i added this comment tagged for 2.13: #5831 |
||
core/server->>core/server/core_app: request to default route<br>(via `http` service) | ||
core/server/core_app->>core/server: call renderCoreApp() | ||
core/server->>core/server/rendering: call render() | ||
critical Initial page bootstrap | ||
core/server/rendering->>core/server/rendering: get theme settings from config | ||
core/server/rendering->>core/server/rendering: assign branding values \<br>(including dark mode) | ||
core/server/rendering->>Browser: return static loading page template | ||
Note over core/server/rendering,Browser: includes inlined font-face styles and static loading page styles | ||
critical <head> (render blocking) | ||
Browser->>Browser: define injection points | ||
Browser->>Browser: load static loading page styles | ||
Browser->>Browser: load font-face styles | ||
Browser->>legacy: load startup.js special route | ||
legacy->>legacy: build startup.js from template | ||
Note over legacy: inject theme settings and font sources | ||
legacy->>Browser: startup.js | ||
critical startup.js | ||
Browser->>Browser: get theme preferences from local storage | ||
Browser->>Browser: set global theme tag | ||
Browser->>Browser: inject theme-specific loading page styles | ||
Browser->>Browser: inject theme-specific font css vars | ||
end | ||
end | ||
Browser->>Browser: render loading/error page<br>(with loaders hidden) | ||
Browser->>legacy: load bootstrap.js special route | ||
legacy->>legacy: build bootstrap.js from template | ||
legacy->>Browser: bootstrap.js | ||
critical bootstrap.js | ||
Browser->>Browser: toggle visibility of errors/loaders | ||
Browser->>Browser: get theme preferences from local storage | ||
Browser->>core/server/core_app: load js bundles | ||
core/server/core_app->>Browser: (React application) | ||
Browser->>core/server/core_app: load theme-specific stylesheets<br>(base, OUI, KUI, legacy) | ||
core/server/core_app->>Browser: themed css | ||
end | ||
end | ||
``` | ||
|
||
### Loading | ||
|
||
`src/legacy/ui/ui_render/ui_render_mixin.js` via `src/legacy/ui/ui_render/bootstrap/template.js.hbs` and `src/legacy/ui/ui_render/bootstrap/app_bootstrap.js`. Aliased in `src/legacy/ui/ui_mixin.js`, called by `src/legacy/server/osd_server.js`. Called by `src/core/server/legacy/legacy_service.ts` via `src/core/server/server.ts` | ||
|
||
### Injected style tags | ||
|
||
1. `src/core/server/rendering/views/styles.tsx` - depends on dark/light mode and injects style tag in head | ||
2. `src/core/server/rendering/views/fonts.tsx` - depends on theme version and injects font style tag in head | ||
3. Monaco editor styles | ||
4. Ace styles | ||
5. Ace TM overrides | ||
6. Ace error styles | ||
6. Component styles | ||
|
||
### Styleshsheets loaded | ||
|
||
Each of the following are loaded in the browser by the [bootstrap script](/src/legacy/ui/ui_render/bootstrap/template.js.hbs) in this order. Currently, these are never unloaded. | ||
|
||
1. Monaco editor styles (e.g. [/packages/osd-ui-shared-deps/target/osd-ui-shared-deps.css](/packages/osd-ui-shared-deps/target/osd-ui-shared-deps.css)), packaged by [/packages/osd-ui-shared-deps/webpack.config.js](/packages/osd-ui-shared-deps/webpack.config.js). In theory, this file could include styles from other shared dependencies, but currently `osd-monaco` is the only package that exports styles. Note that these are the default, un-themed styles; theming of monaco editors is handled by [/src/plugins/opensearch_dashboards_react/public/code_editor/editor_theme.ts](/src/plugins/opensearch_dashboards_react/public/code_editor/editor_theme.ts). | ||
2. Theme and mode-specific OUI styles (e.g. [](), compiled by `packages/osd-ui-shared-deps/webpack.config.js`). | ||
3. Theme and mode-specific KUI styles (e.g. `packages/osd-ui-framework/src/kui_next_dark.scss`, compiled by `packages/osd-ui-framework/Gruntfile.js`). Separate stylesheets for each theme version/dark mode combo (colors). | ||
4. Mode-specific legacy styles (e.g. [/src/core/server/core_app/assets/legacy_dark_theme.css](/src/core/server/core_app/assets/legacy_dark_theme.css)) | ||
|
||
Component styles are not loaded as stylesheets. | ||
|
||
## Current theme usage | ||
|
||
### JSON/JS Vars | ||
|
||
1. Defined by `packages/osd-ui-shared-deps/theme.ts` | ||
1. Used by `src/plugins/charts/public/static/color_maps/color_maps.ts` to set vis colors | ||
2. Used by `src/plugins/discover/public/application/components/chart/histogram/histogram.tsx` to define Discover histogram Elastic Chart styling | ||
3. Used by `src/plugins/maps_legacy/public/map/opensearch_dashboards_map.js` and `src/plugins/region_map/public/choropleth_layer.js` for minor map UI styling (line color, empty shade) | ||
4. Used by `src/plugins/vis_type_vega/public/data_model/vega_parser.ts` for Vega/Vega-Lite theming | ||
2. Used by `src/plugins/vis_type_vislib/public/vislib/components/tooltip/tooltip.js` for tooltip spacing | ||
3. Used by `src/plugins/expressions/public/react_expression_renderer.tsx` to define padding options. | ||
4. Used by `src/core/server/rendering/views/theme.ts` to inject values into `src/core/server/rendering/views/styles.tsx` | ||
5. Used (incorrectly) to style a badge color in `src/plugins/index_pattern_management/public/components/create_button/create_button.tsx` | ||
6. Used by `src/plugins/opensearch_dashboards_react/public/code_editor/editor_theme.ts` to create Monaco theme styles |
AMoo-Miki marked this conversation as resolved.
Show resolved
Hide resolved
|
Original file line number | Diff line number | Diff line change | ||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -56,7 +56,12 @@ | |||||||||||||||||
constructor(params: UiSettingsClientParams) { | ||||||||||||||||||
this.api = params.api; | ||||||||||||||||||
this.defaults = cloneDeep(params.defaults); | ||||||||||||||||||
this.cache = defaultsDeep({}, this.defaults, cloneDeep(params.initialSettings)); | ||||||||||||||||||
this.cache = defaultsDeep( | ||||||||||||||||||
{}, | ||||||||||||||||||
this.defaults, | ||||||||||||||||||
cloneDeep(params.initialSettings), | ||||||||||||||||||
this.getBrowserStoredSettings() | ||||||||||||||||||
); | ||||||||||||||||||
|
||||||||||||||||||
params.done$.subscribe({ | ||||||||||||||||||
complete: () => { | ||||||||||||||||||
|
@@ -173,6 +178,28 @@ | |||||||||||||||||
return this.updateErrors$.asObservable(); | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
private getBrowserStoredSettings() { | ||||||||||||||||||
const uiSettingsJSON = window.localStorage.getItem('uiSettings') || '{}'; | ||||||||||||||||||
try { | ||||||||||||||||||
return JSON.parse(uiSettingsJSON); | ||||||||||||||||||
} catch (error) { | ||||||||||||||||||
this.updateErrors$.next(error); | ||||||||||||||||||
} | ||||||||||||||||||
return {}; | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
private setBrowserStoredSettings(key: string, newVal: any) { | ||||||||||||||||||
const oldSettings = this.getBrowserStoredSettings(); | ||||||||||||||||||
const newSettings = cloneDeep(oldSettings); | ||||||||||||||||||
if (newVal === null) { | ||||||||||||||||||
delete newSettings[key]; | ||||||||||||||||||
} else { | ||||||||||||||||||
newSettings[key] = { userValue: newVal }; | ||||||||||||||||||
} | ||||||||||||||||||
window.localStorage.setItem(`uiSettings`, JSON.stringify(newSettings)); | ||||||||||||||||||
return { settings: newSettings }; | ||||||||||||||||||
BSFishy marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
private assertUpdateAllowed(key: string) { | ||||||||||||||||||
if (this.isOverridden(key)) { | ||||||||||||||||||
throw new Error( | ||||||||||||||||||
|
@@ -198,8 +225,10 @@ | |||||||||||||||||
this.setLocally(key, newVal); | ||||||||||||||||||
|
||||||||||||||||||
try { | ||||||||||||||||||
const { settings } = (await this.api.batchSet(key, newVal)) || {}; | ||||||||||||||||||
this.cache = defaultsDeep({}, defaults, settings); | ||||||||||||||||||
const { settings } = this.cache[key]?.preferBrowserSetting | ||||||||||||||||||
? this.setBrowserStoredSettings(key, newVal) | ||||||||||||||||||
: (await this.api.batchSet(key, newVal)) || {}; | ||||||||||||||||||
this.cache = defaultsDeep({}, defaults, this.getBrowserStoredSettings(), settings); | ||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Josh, can you help me understand this please?
Result: The cache will contain There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That's the correct interpretation, and the intended behavior.
Yep, and this is coming from the configs - under this new implementation, there's no longer the UI ability to set a cluster-wide default.
This setting is irrelevant for the purpose of this client and not read. It's only purpose is to change/update the localStorage settings on bootstrap/startup
This is the source of truth for any UISetting that has the
Yep - in the current implementation, we wouldn't look there. We may change this in the future so that we use/store a persistent source of truth there which can be utilized by the bootstrap/startup initialization; but the source of truth would remain the local storage value. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @joshuarrrr can you write up a short readme about how UI settings will behave after this change from the perspective of a developer who is writing UI settings. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
@joshuarrrr, when
I am not sure i follow. This is the only place I found any logic attached to The line updating
Suggested change
Should There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. is this a blocker @AMoo-Miki ? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Rocky, I think the unexpected result will confuse users enough to make it a problem. If i find 2 more people that agree with me, we would change this and move forward without bugging Josh. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. PS, i could make the change myself but I want Josh to challenge my expectation to make sure we make the best offering for the users. |
||||||||||||||||||
this.saved$.next({ key, newValue: newVal, oldValue: initialVal }); | ||||||||||||||||||
return true; | ||||||||||||||||||
} catch (error) { | ||||||||||||||||||
|
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
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 dont totally understand the theme system even after reading this, but i think that has less to do with your writing and more to do with how complex and confusing it is. Cannot tell you how much i appreciate this doc though.