Skip to content

Commit

Permalink
HPCC-27646 Create theme editor
Browse files Browse the repository at this point in the history
Signed-off-by: Jeremy Clements <[email protected]>
  • Loading branch information
jeclrsg committed Dec 1, 2023
1 parent 59f5cd4 commit e050048
Show file tree
Hide file tree
Showing 18 changed files with 2,204 additions and 147 deletions.
2 changes: 1 addition & 1 deletion esp/src/eclwatch/templates/HPCCPlatformWidget.html
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,7 @@
<div data-dojo-type="hpcc.TableContainer">
<input id="${id}EnvironmentTextCB" title="${i18n.EnableBannerText}" data-dojo-type="dijit.form.CheckBox" />
<input id="${id}EnvironmentText" title="${i18n.NameOfEnvironment}" data-dojo-props="placeHolder: '${i18n.NameOfEnvironment}', trim: true" data-dojo-type="dijit.form.TextBox" />
<input id="${id}ToolbarColor" title="${i18n.BannerColor}:" style="width:100%;" value="red" data-dojo-props="trim: true" data-dojo-type="HPCCColorPicker" />
<input id="${id}ToolbarColor" title="${i18n.ToolbarColor}:" style="width:100%;" value="red" data-dojo-props="trim: true" data-dojo-type="HPCCColorPicker" />
</div>
</div>
</div>
Expand Down
20 changes: 12 additions & 8 deletions esp/src/src-react/components/Menu.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import { useFavorite, useFavorites, useHistory } from "../hooks/favorite";
import { useUserTheme } from "../hooks/theme";
import { usePivotItemDisable } from "../layouts/pivot";
import { Breadcrumbs } from "./Breadcrumbs";
import { Settings } from "./Settings";

// Top Level Nav ---
function navLinkGroups(): INavLinkGroup[] {
Expand Down Expand Up @@ -99,29 +100,32 @@ export const MainNavigation: React.FunctionComponent<MainNavigationProps> = ({
}) => {

const menu = useConst([...navLinkGroups()]);
const { theme, setTheme, isDark } = useUserTheme();
const { theme, setThemeDark, isDark } = useUserTheme();

const [showSettings, setShowSettings] = React.useState(false);

const selKey = React.useMemo(() => {
return navSelectedKey(hashPath);
}, [hashPath]);

return <Stack verticalAlign="space-between" styles={{ root: { width: `${FIXED_WIDTH}px`, height: "100%", position: "relative", backgroundColor: theme.palette.themeLighterAlt } }}>
return <Stack verticalAlign="space-between" styles={{ root: { width: `${FIXED_WIDTH}px`, height: "100%", position: "relative", backgroundColor: theme?.palette?.themeLighterAlt ?? "white" } }}>
<Stack.Item>
<Nav selectedKey={selKey} groups={menu} />
</Stack.Item>
<Stack.Item>
<IconButton
iconProps={{ iconName: isDark ? "Sunny" : "ClearNight" }}
onClick={() => {
setTheme(isDark ? "light" : "dark");
// setTheme(isDark ? theme);
setThemeDark(isDark ? "light" : "dark");
const themeChangeEvent = new CustomEvent("eclwatch-theme-toggle", {
detail: { dark: !isDark }
});
document.dispatchEvent(themeChangeEvent);
}}
/>
{/* Disable Theme editor button for launch of 9.0 */}
{/* <IconButton iconProps={{ iconName: "Equalizer" }} onClick={() => { }} /> */}
<IconButton iconProps={{ iconName: "Equalizer" }} onClick={() => setShowSettings(true)} />
<Settings show={showSettings} onClose={() => setShowSettings(false)} />
</Stack.Item>
</Stack>;
};
Expand Down Expand Up @@ -309,9 +313,9 @@ export const SubNavigation: React.FunctionComponent<SubNavigationProps> = ({
{!subNav &&
<Stack.Item grow={1} style={{ lineHeight: "24px" }}>
{hashPath.includes("/files/")
? <Breadcrumbs hashPath={hashPath} ignoreN={2} />
: <Breadcrumbs hashPath={hashPath} ignoreN={1} />
}
? <Breadcrumbs hashPath={hashPath} ignoreN={2} />
: <Breadcrumbs hashPath={hashPath} ignoreN={1} />
}
</Stack.Item>
}
</Stack>
Expand Down
69 changes: 69 additions & 0 deletions esp/src/src-react/components/Settings.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
import * as React from "react";
import { DefaultButton, Pivot, PivotItem, PrimaryButton } from "@fluentui/react";
import { ThemeEditor } from "./ThemeEditor";
import { useUserTheme } from "../hooks/theme";
import { MessageBox } from "../layouts/MessageBox";
import nlsHPCC from "src/nlsHPCC";

interface SettingsProps {
show?: boolean;
onClose?: () => void;
}

export const Settings: React.FunctionComponent<SettingsProps> = ({
show = false,
onClose = () => { }
}) => {

const { primaryColor, setPrimaryColor, resetTheme } = useUserTheme();
const [previousColor, setPreviousColor] = React.useState(primaryColor);
const [activePivot, setActivePivot] = React.useState("theme");

React.useEffect(() => {
// cache the previous color at dialog open, used to reset upon close
if (show) setPreviousColor(primaryColor);
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [show]);

return <MessageBox show={show} setShow={onClose} blocking={true} modeless={false} title={nlsHPCC.Settings} minWidth={400}
footer={<>
<PrimaryButton onClick={() => {
if (onClose) {
onClose();
}
}} text={nlsHPCC.OK} />
<DefaultButton onClick={() => {
switch (activePivot) {
case "theme":
default:
setPrimaryColor(previousColor);
break;

}
if (onClose) {
onClose();
}
}} text={nlsHPCC.Cancel} />
<DefaultButton onClick={() => {
switch (activePivot) {
case "theme":
default:
resetTheme();
break;

}
if (onClose) {
onClose();
}
}} text={nlsHPCC.Reset} />
</>}>
<Pivot onLinkClick={item => {
setActivePivot(item.props.itemKey);
}}>
<PivotItem itemKey="theme" headerText={nlsHPCC.Theme}>
<ThemeEditor />
</PivotItem>
</Pivot>
</MessageBox>;

};
108 changes: 108 additions & 0 deletions esp/src/src-react/components/ThemeEditor.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
import * as React from "react";
import { Callout, ColorPicker, getColorFromString, IColor, Label, mergeStyles, Stack, TextField } from "@fluentui/react";
import { useUserTheme } from "../hooks/theme";
import nlsHPCC from "src/nlsHPCC";

const colorBoxClassName = mergeStyles({
width: 20,
height: 20,
display: "inline-block",
position: "absolute",
left: 5,
top: 5,
border: "1px solid black",
flexShrink: 0,
});

const textBoxClassName = mergeStyles({
width: 100,
});

const colorPanelClassName = mergeStyles({
position: "relative",
});

interface ThemeEditorColorPickerProps {
color: IColor;
onColorChange: (color: IColor | undefined) => void;
label: string;
}

export const ThemeEditorColorPicker: React.FunctionComponent<ThemeEditorColorPickerProps> = ({
color,
onColorChange,
label
}) => {
const [pickerColor, setPickerColor] = React.useState(color);
const _colorPickerRef = React.useRef(null);
const [editingColorStr, setEditingColorStr] = React.useState(pickerColor?.str ?? "");
const [isColorPickerVisible, setIsColorPickerVisible] = React.useState<boolean>(false);

const _onTextFieldValueChange = React.useCallback((ev: any, newValue: string | undefined) => {
const newColor = getColorFromString(newValue);
if (newColor) {
onColorChange(newColor);
setEditingColorStr(newColor.str);
} else {
setEditingColorStr(newValue);
}
}, [onColorChange, setEditingColorStr]);

React.useEffect(() => {
setPickerColor(color);
}, [color]);

React.useEffect(() => {
setEditingColorStr(pickerColor?.str);
}, [pickerColor]);

return <div>
<Stack horizontal horizontalAlign={"space-between"} tokens={{ childrenGap: 20 }}>
<Label>{label}</Label>
<Stack horizontal className={colorPanelClassName} tokens={{ childrenGap: 35 }}>
<div
ref={_colorPickerRef}
id="colorbox"
className={colorBoxClassName}
style={{ backgroundColor: color?.str }}
onClick={() => setIsColorPickerVisible(true)}
/>
<TextField
id="textfield"
className={textBoxClassName}
value={editingColorStr}
onChange={_onTextFieldValueChange}
/>
</Stack>
</Stack>
{isColorPickerVisible && (
<Callout
gapSpace={10}
target={_colorPickerRef.current}
setInitialFocus={true}
onDismiss={() => setIsColorPickerVisible(false)}
>
<ColorPicker color={pickerColor} onChange={(evt, color) => onColorChange(color)} alphaType={"none"} />
</Callout>
)}
</div>;
};

interface ThemeEditorProps {
}

export const ThemeEditor: React.FunctionComponent<ThemeEditorProps> = () => {

const { primaryColor, setPrimaryColor } = useUserTheme();

return <div style={{ minHeight: "208px", paddingTop: "32px" }}>
<ThemeEditorColorPicker
color={getColorFromString(primaryColor)}
onColorChange={(newColor: IColor | undefined) => {
setPrimaryColor(newColor.str);
}}
label={nlsHPCC.PrimaryColor}
/>
</div>;

};
Loading

0 comments on commit e050048

Please sign in to comment.