diff --git a/.eslintrc b/.eslintrc index 206609861e01..992dfe343fa3 100644 --- a/.eslintrc +++ b/.eslintrc @@ -105,6 +105,7 @@ "nonpersistent", "noopener", "noreferrer", + "nowrap", "num", "numpad", "operatingsystem", diff --git a/app/assets/stylesheets/base.scss b/app/assets/stylesheets/base.scss index 35834c1662c9..4cd4a168e087 100644 --- a/app/assets/stylesheets/base.scss +++ b/app/assets/stylesheets/base.scss @@ -5,14 +5,26 @@ html { } #rails-app-content { + --header-height: 70px; position: absolute; - top: 70px; + top: var(--header-height); left: 0; right: 0; bottom: 0; overflow: auto; - height: calc(100% - 70px); + height: calc(100% - var(--header-height)); margin-left: 250px; + &.user-banner-present { + --banner-height: calc( + 2 * var(--pf-global--spacer--xs) + + (var(--pf-global--LineHeight--md) * var(--pf-global--FontSize--sm)) + ); // banner height is line height and a small padding + top: calc( + var(--header-height) + var(--banner-height) + ); + + height: calc(100% - var(--header-height) - var(--banner-height)); + } .rails-table-toolbar { padding-bottom: 0; display: flex; @@ -24,7 +36,6 @@ html { grid-template-columns: unset; grid-template-areas: unset; } - } body { diff --git a/app/helpers/layout_helper.rb b/app/helpers/layout_helper.rb index 8ca268df7ce9..bdef520fc678 100644 --- a/app/helpers/layout_helper.rb +++ b/app/helpers/layout_helper.rb @@ -51,7 +51,8 @@ def layout_data user: fetch_user, brand: 'foreman', root: main_app.root_path, locations: fetch_locations, orgs: fetch_organizations, - instance_title: Setting[:instance_title] + instance_title: Setting[:instance_title], + instance_color: Setting[:instance_color] } end diff --git a/app/registries/foreman/settings/general.rb b/app/registries/foreman/settings/general.rb index bb2c86a2c409..17c16ea7824b 100644 --- a/app/registries/foreman/settings/general.rb +++ b/app/registries/foreman/settings/general.rb @@ -86,9 +86,14 @@ collection: timezones) setting('instance_title', type: :string, - description: N_("The instance title is shown on the top navigation bar (requires a page reload)."), + description: N_("The instance title is shown above the header in a banner (requires a page reload)."), default: nil, full_name: N_('Instance title')) + setting('instance_color', + type: :string, + description: N_("Hex value (Example: #000000) for color for the instance title banner. Will only be used if an instance title is defined. (requires a page reload)."), + default: '#000000', + full_name: N_('Instance color')) setting('audits_period', type: :integer, description: N_('Duration in days to preserve audits for. Leave empty to disable the audits cleanup.'), diff --git a/app/views/layouts/base.html.erb b/app/views/layouts/base.html.erb index c1f0f205bd02..51f555a29469 100644 --- a/app/views/layouts/base.html.erb +++ b/app/views/layouts/base.html.erb @@ -59,7 +59,7 @@ <% end %>
<%= yield(:content) %> diff --git a/webpack/assets/javascripts/react_app/components/Layout/Layout.js b/webpack/assets/javascripts/react_app/components/Layout/Layout.js index cfa0c73d7691..a740ad334a71 100644 --- a/webpack/assets/javascripts/react_app/components/Layout/Layout.js +++ b/webpack/assets/javascripts/react_app/components/Layout/Layout.js @@ -1,7 +1,8 @@ import React from 'react'; -import { Page, PageSidebar } from '@patternfly/react-core'; +import { Page, PageSidebar, Flex, FlexItem } from '@patternfly/react-core'; import { layoutPropTypes, layoutDefaultProps } from './LayoutHelper'; +import { InstanceBanner } from './components/InstanceBanner'; import Header from './components/Toolbar/Header'; import Navigation from './Navigation'; import './layout.scss'; @@ -29,28 +30,44 @@ const Layout = ({ }; return ( <> - - } - id="foreman-page" - sidebar={ - + + + + + } - /> - } - > - {children} - + id="foreman-page" + sidebar={ + + } + /> + } + > + {children} + + + ); }; diff --git a/webpack/assets/javascripts/react_app/components/Layout/__tests__/Layout.test.js b/webpack/assets/javascripts/react_app/components/Layout/__tests__/Layout.test.js index 780b2e6fd39b..0f0ff7e619b0 100644 --- a/webpack/assets/javascripts/react_app/components/Layout/__tests__/Layout.test.js +++ b/webpack/assets/javascripts/react_app/components/Layout/__tests__/Layout.test.js @@ -31,5 +31,6 @@ describe('Layout', () => { expect(screen.getByText('Monitor')).toBeVisible(); expect(screen.getByText('Dashboard')).not.toBeVisible(); expect(screen.getByText('All Hosts')).toBeVisible(); + expect(screen.getByText('Production')).toBeVisible(); }); }); diff --git a/webpack/assets/javascripts/react_app/components/Layout/components/InstanceBanner.js b/webpack/assets/javascripts/react_app/components/Layout/components/InstanceBanner.js new file mode 100644 index 000000000000..6c759b0e1b0d --- /dev/null +++ b/webpack/assets/javascripts/react_app/components/Layout/components/InstanceBanner.js @@ -0,0 +1,69 @@ +import React from 'react'; +import PropTypes from 'prop-types'; +import { Banner } from '@patternfly/react-core'; + +const getContrastColor = backgroundColor => { + const hexToRgb = hex => { + const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex); + return result + ? { + r: parseInt(result[1], 16), + g: parseInt(result[2], 16), + b: parseInt(result[3], 16), + } + : null; + }; + backgroundColor = hexToRgb(backgroundColor); + // Calculate the relative luminance of the background color + const luminance = + (0.2126 * backgroundColor.r + + 0.7152 * backgroundColor.g + + 0.0722 * backgroundColor.b) / + 255; + + // Choose black or white text based on the relative luminance + return luminance > 0.5 ? 'black' : 'white'; +}; +const validateHexColor = instanceColor => { + // Check if the string is a valid hex color code + if (/^#([0-9A-Fa-f]{3}){1,2}$/.test(instanceColor)) { + return instanceColor; + } + return '#000000'; +}; + +export const InstanceBanner = ({ data }) => { + if (!data || !data.instance_title) { + return null; + } + const instance = data.instance_title; + const instanceColor = validateHexColor(data.instance_color); + return ( + instance && ( + +
+ {instance} +
+
+ ) + ); +}; + +InstanceBanner.propTypes = { + data: PropTypes.shape({ + instance_title: PropTypes.string, + instance_color: PropTypes.string, + }), +}; + +InstanceBanner.defaultProps = { + data: {}, +}; diff --git a/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/HeaderToolbar.js b/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/HeaderToolbar.js index e0346ddc87b1..0f1674cf025f 100644 --- a/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/HeaderToolbar.js +++ b/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/HeaderToolbar.js @@ -17,7 +17,6 @@ import { organizationPropType, userPropType, } from '../../LayoutHelper'; -import InstanceTitleViewer from './InstanceTitleViewer'; import './HeaderToolbar.scss'; const HeaderToolbar = ({ @@ -26,7 +25,6 @@ const HeaderToolbar = ({ notification_url: notificationUrl, user, stop_impersonation_url: stopImpersonationUrl, - instance_title: instanceTitle, isLoading, }) => ( @@ -39,9 +37,6 @@ const HeaderToolbar = ({ /> - - - @@ -60,7 +55,6 @@ const HeaderToolbar = ({ ); HeaderToolbar.propTypes = { stop_impersonation_url: PropTypes.string.isRequired, - instance_title: PropTypes.string, locations: locationPropType.isRequired, orgs: organizationPropType.isRequired, notification_url: PropTypes.string.isRequired, @@ -69,7 +63,6 @@ HeaderToolbar.propTypes = { }; HeaderToolbar.defaultProps = { - instance_title: null, user: {}, isLoading: layoutDefaultProps.isLoading, }; diff --git a/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/InstanceTitleViewer.js b/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/InstanceTitleViewer.js deleted file mode 100644 index cd6e89539b4e..000000000000 --- a/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/InstanceTitleViewer.js +++ /dev/null @@ -1,29 +0,0 @@ -import React from 'react'; -import { Icon } from 'patternfly-react'; -import { Tooltip, TooltipPosition } from '@patternfly/react-core'; -import PropTypes from 'prop-types'; - -const InstanceTitleViewer = ({ title }) => { - if (!title) { - return null; - } - - return ( - - - - ); -}; - -InstanceTitleViewer.propTypes = { - /** Title to display */ - title: PropTypes.string, -}; -InstanceTitleViewer.defaultProps = { - title: '', -}; -export default InstanceTitleViewer; diff --git a/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/__snapshots__/HeaderToolbar.test.js.snap b/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/__snapshots__/HeaderToolbar.test.js.snap index 324a0762eaf5..cba23cba6207 100644 --- a/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/__snapshots__/HeaderToolbar.test.js.snap +++ b/webpack/assets/javascripts/react_app/components/Layout/components/Toolbar/__snapshots__/HeaderToolbar.test.js.snap @@ -53,11 +53,6 @@ exports[`HeaderToolbar rendering render HeaderToolbar 1`] = ` } } > - - -