diff --git a/app/front-end/index.html b/app/front-end/index.html index 9d361df..d0fc71b 100644 --- a/app/front-end/index.html +++ b/app/front-end/index.html @@ -10,10 +10,18 @@ font-family: 'Nunito', sans-serif; } - - + + Kath @@ -21,34 +29,34 @@
+ // Set background color based on system color mode + setBackgroundColor(systemColorMode); + } + diff --git a/app/front-end/public/Kath_Favicon.svg b/app/front-end/public/Kath_Favicon.svg new file mode 100644 index 0000000..dbc68fd --- /dev/null +++ b/app/front-end/public/Kath_Favicon.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/app/front-end/public/Kath_OnlyLogo.svg b/app/front-end/public/Kath_OnlyLogo.svg deleted file mode 100644 index e499051..0000000 --- a/app/front-end/public/Kath_OnlyLogo.svg +++ /dev/null @@ -1,20 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/app/front-end/src/components/buttons/IconTitleButton.tsx b/app/front-end/src/components/buttons/IconTitleButton.tsx index 3c72a19..4f9d877 100644 --- a/app/front-end/src/components/buttons/IconTitleButton.tsx +++ b/app/front-end/src/components/buttons/IconTitleButton.tsx @@ -1,5 +1,5 @@ import { Colors } from '@/types'; -import { Box, IconButton, Typography } from '@mui/material'; +import { alpha, Box, IconButton, Typography, useTheme } from '@mui/material'; interface Props { icon: React.ReactNode; @@ -30,6 +30,7 @@ interface Props { * @returns {JSX.Element} The `IconTitleButton` component rendering an icon button and an optional title. */ export const IconTitleButton: React.FC = ({ icon, title, width, height, borderRadius, onClick }) => { + const Theme = useTheme(); return ( = ({ icon, title, width, height, b height: height || '3rem', borderRadius: borderRadius || '1rem', transition: 'background-color 0.5s ease', - ':hover': { backgroundColor: 'rgba(216, 228, 232, 0.5)' }, // TODO: fix this to be responsive to theme + ':hover': { + backgroundColor: alpha(Theme.palette.primary.contrastText, 0.2), + }, }} onClick={onClick} > diff --git a/app/front-end/src/components/dialogs/settingsDialog/index.ts b/app/front-end/src/components/dialogs/settingsDialog/index.ts new file mode 100644 index 0000000..4a1faa8 --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/index.ts @@ -0,0 +1,3 @@ +export { SettingsDialog } from './settingsDialog'; +export { SettingSpacer } from './settingSpacer'; +export { SettingsSelectField } from './settingsSelectField'; diff --git a/app/front-end/src/components/modals/settingsDialog/settingSpacer.tsx b/app/front-end/src/components/dialogs/settingsDialog/settingSpacer.tsx similarity index 100% rename from app/front-end/src/components/modals/settingsDialog/settingSpacer.tsx rename to app/front-end/src/components/dialogs/settingsDialog/settingSpacer.tsx diff --git a/app/front-end/src/components/dialogs/settingsDialog/settingsDialog.tsx b/app/front-end/src/components/dialogs/settingsDialog/settingsDialog.tsx new file mode 100644 index 0000000..7f56ed1 --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/settingsDialog.tsx @@ -0,0 +1,62 @@ +import { SettingSpacer } from '@/components/dialogs/settingsDialog'; +import { ColorModeSetting, LanguageSetting, TimeZoneSetting } from '@/components/dialogs/settingsDialog/settingsFields'; +import { Close as CloseIcon } from '@mui/icons-material'; +import { Box, Dialog, DialogContent, DialogTitle, Grid, IconButton, styled, useTheme } from '@mui/material'; + +const BootstrapDialog = styled(Dialog)(({ theme }) => ({ + backdropFilter: 'blur(5px)', + '& .MuiDialogContent-root': { + padding: '1.5rem', + }, + '& .MuiDialogActions-root': { + padding: '1.5rem', + }, + '& .MuiDialog-paper': { + borderRadius: '1.5rem', + minWidth: '25%', + backgroundColor: theme.palette.background.default, + }, +})); + +interface SettingsDialogProps { + open: boolean; + handleClose: () => void; +} + +export const SettingsDialog: React.FC = ({ open, handleClose }) => { + const Theme = useTheme(); + + return ( + + + + + Settings + + + + + + + + + + + + + + + + + + + ); +}; diff --git a/app/front-end/src/components/dialogs/settingsDialog/settingsFields/colorModeSetting.tsx b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/colorModeSetting.tsx new file mode 100644 index 0000000..6eeef5b --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/colorModeSetting.tsx @@ -0,0 +1,23 @@ +import { SettingsSelectField } from '@/components/dialogs/settingsDialog'; +import { useThemeContext } from '@/hooks'; + +export const ColorModeSetting = () => { + const ThemeContext = useThemeContext(); + + const handleThemeChange = () => { + ThemeContext.update(); + }; + + return ( + + ); +}; diff --git a/app/front-end/src/components/dialogs/settingsDialog/settingsFields/index.ts b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/index.ts new file mode 100644 index 0000000..403bde7 --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/index.ts @@ -0,0 +1,3 @@ +export { ColorModeSetting } from './colorModeSetting'; +export { LanguageSetting } from './languageSetting'; +export { TimeZoneSetting } from './timeZoneSetting'; diff --git a/app/front-end/src/components/dialogs/settingsDialog/settingsFields/languageSetting.tsx b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/languageSetting.tsx new file mode 100644 index 0000000..44a1813 --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/languageSetting.tsx @@ -0,0 +1,28 @@ +import { SettingsSelectField } from '@/components/dialogs/settingsDialog'; +import { SelectChangeEvent } from '@mui/material'; +import { useState } from 'react'; + +export const LanguageSetting = () => { + // TODO: Implement language switching functionality and replace with correct values + + const [switchLanguage, setSwitchLanguage] = useState('en'); + + const handleLanguageChange = (event: SelectChangeEvent) => { + const selectedLanguage = event.target.value; + setSwitchLanguage(selectedLanguage); + }; + + return ( + + ); +}; diff --git a/app/front-end/src/components/dialogs/settingsDialog/settingsFields/timeZoneSetting.tsx b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/timeZoneSetting.tsx new file mode 100644 index 0000000..88a270c --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/settingsFields/timeZoneSetting.tsx @@ -0,0 +1,24 @@ +import { SettingsSelectField } from '@/components/dialogs/settingsDialog'; +import { SelectChangeEvent } from '@mui/material'; +import { useState } from 'react'; + +export const TimeZoneSetting = () => { + // TODO: Implement time zone switching functionality and replace with correct values + + const [switchTimeZone, setSwitchTimeZone] = useState('GMT+3'); + + const handleTimeZoneChange = (event: SelectChangeEvent) => { + const selectedValue = event.target.value; + setSwitchTimeZone(selectedValue); + }; + + return ( + + ); +}; diff --git a/app/front-end/src/components/dialogs/settingsDialog/settingsSelectField.tsx b/app/front-end/src/components/dialogs/settingsDialog/settingsSelectField.tsx new file mode 100644 index 0000000..7196f33 --- /dev/null +++ b/app/front-end/src/components/dialogs/settingsDialog/settingsSelectField.tsx @@ -0,0 +1,53 @@ +import { Box, FormControl, Grid, MenuItem, Select, SelectChangeEvent, Typography, useTheme } from '@mui/material'; +import React from 'react'; + +interface Setting { + value: string | undefined; + label: string; +} + +interface SettingsSelectFieldProps { + title: string; + description: string; + settings: Setting[]; + value: string | undefined; + onChange: (event: SelectChangeEvent) => void; +} + +export const SettingsSelectField: React.FC = ({ + title, + description, + settings, + value, + onChange, +}) => { + const Theme = useTheme(); + + return ( + + + {title} + {description} + + + + + + + + + + ); +}; diff --git a/app/front-end/src/components/layouts/baseLayout.tsx b/app/front-end/src/components/layouts/baseLayout.tsx index 109e50e..5e22872 100644 --- a/app/front-end/src/components/layouts/baseLayout.tsx +++ b/app/front-end/src/components/layouts/baseLayout.tsx @@ -1,7 +1,12 @@ import { IconTitleButton } from '@/components/buttons/IconTitleButton'; -import SettingsDialog from '@/components/modals/settingsDialog/settingsDialog'; +import { SettingsDialog } from '@/components/dialogs/settingsDialog'; import { Colors } from '@/types'; -import { AutoMode, Home, SettingsOutlined, SwitchAccessShortcut } from '@mui/icons-material'; +import { + AutoMode as AutoModeIcon, + Home as HomeIcon, + SettingsOutlined as SettingsOutlinedIcon, + SwitchAccessShortcut as SwitchAccessShortcutIcon, +} from '@mui/icons-material'; import { Box, Typography, useTheme } from '@mui/material'; import { useState } from 'react'; @@ -37,7 +42,14 @@ export const BaseLayout: React.FC = ({ children }) => { }; return ( - + = ({ children }) => { Version 1.0.02 {/* TODO: add application context provider to get values of it */} - + = ({ children }) => { }} > } + icon={} title={'Home'} /> } + icon={} title={'Macros'} /> @@ -122,13 +134,15 @@ export const BaseLayout: React.FC = ({ children }) => { > + } onClick={handleOptionsMenuOpen} /> } @@ -136,9 +150,9 @@ export const BaseLayout: React.FC = ({ children }) => { - {children} + ); diff --git a/app/front-end/src/features/editor/components/editorView/editorToolbar.tsx b/app/front-end/src/features/editor/components/editorView/editorToolbar.tsx index 1fcfe40..61c17d1 100644 --- a/app/front-end/src/features/editor/components/editorView/editorToolbar.tsx +++ b/app/front-end/src/features/editor/components/editorView/editorToolbar.tsx @@ -1,5 +1,4 @@ import { socket } from '@/lib'; -import { Colors } from '@/types'; import { Done as DoneIcon, Error as ErrorIcon } from '@mui/icons-material'; import { Box, Button, CircularProgress, useTheme } from '@mui/material'; import { @@ -78,18 +77,25 @@ export const EditorToolbar: React.FC = ({ disabled, handleSa setIsSaving(true); handleSave(); }} - disabled={disabled} + disabled={disabled || isSaving} startIcon={ isSaving ? ( - + ) : saveStatus ? ( - + ) : ( ) } size='small' - variant='contained' + variant='outlined' + sx={{ + color: saveStatus === false ? Theme.palette.error.main : Theme.palette.primary.main, + borderColor: saveStatus === false ? Theme.palette.error.main : Theme.palette.primary.main, + '&:hover': { + borderColor: saveStatus === false ? Theme.palette.error.dark : Theme.palette.primary.dark, + }, + }} > Save diff --git a/app/front-end/src/features/editor/components/fileTreeView/fileTreeItem/fileTreeItem.tsx b/app/front-end/src/features/editor/components/fileTreeView/fileTreeItem/fileTreeItem.tsx index 332636d..78fb186 100644 --- a/app/front-end/src/features/editor/components/fileTreeView/fileTreeItem/fileTreeItem.tsx +++ b/app/front-end/src/features/editor/components/fileTreeView/fileTreeItem/fileTreeItem.tsx @@ -16,7 +16,8 @@ import React from 'react'; import { FileTreeItemLabel } from '.'; const StyledFileTreeItemRoot = styled(TreeItem2Root)(({ theme }) => ({ - color: theme.palette.mode === 'light' ? theme.palette.grey[800] : theme.palette.grey[400], + //color: theme.palette.mode === 'light' ? theme.palette.grey[800] : theme.palette.grey[400], + color: theme.palette.text.secondary, position: 'relative', [`& .${treeItemClasses.groupTransition}`]: { marginLeft: theme.spacing(3.5), @@ -33,7 +34,7 @@ const StyledFileTreeItemContent = styled(TreeItem2Content)(({ theme }) => ({ fontWeight: 500, ['&.Mui-expanded ']: { '&:not(.Mui-focused, .Mui-selected, .Mui-selected.Mui-focused) .labelIcon': { - color: theme.palette.mode === 'light' ? theme.palette.primary.main : theme.palette.primary.dark, + color: theme.palette.primary.main, }, '&::before': { content: '""', @@ -43,15 +44,16 @@ const StyledFileTreeItemContent = styled(TreeItem2Content)(({ theme }) => ({ top: '44px', height: 'calc(100% - 48px)', width: '1.5px', - backgroundColor: theme.palette.mode === 'light' ? theme.palette.grey[300] : theme.palette.grey[700], + backgroundColor: theme.palette.secondary.main, }, }, '&:hover': { backgroundColor: alpha(theme.palette.primary.main, 0.1), - color: theme.palette.mode === 'light' ? theme.palette.primary.main : 'white', + color: theme.palette.primary.main, }, ['&.Mui-focused, &.Mui-selected, &.Mui-selected.Mui-focused']: { - backgroundColor: theme.palette.mode === 'light' ? theme.palette.primary.main : theme.palette.primary.dark, + backgroundColor: + theme.palette.mode === 'light' ? alpha(theme.palette.primary.main, 0.7) : alpha(theme.palette.primary.main, 0.5), color: theme.palette.primary.contrastText, }, })); diff --git a/app/front-end/src/features/editor/components/filebarView/filebarGroupItem.tsx b/app/front-end/src/features/editor/components/filebarView/filebarGroupItem.tsx index bf7c334..8ce0086 100644 --- a/app/front-end/src/features/editor/components/filebarView/filebarGroupItem.tsx +++ b/app/front-end/src/features/editor/components/filebarView/filebarGroupItem.tsx @@ -1,7 +1,7 @@ import { useWorkspaceContext } from '@/features/editor/hooks'; import { FileTypes } from '@/types'; import { Close as CloseIcon } from '@mui/icons-material'; -import { Box, IconButton, Typography, useTheme } from '@mui/material'; +import { alpha, Box, IconButton, Typography, useTheme } from '@mui/material'; export interface FilebarGroupItemProps { fileId: string; @@ -40,9 +40,12 @@ export const FilebarGroupItem: React.FC = ({ fileId, file pl: '1rem', pr: '0.5rem', bgcolor: Workspace.fileId === fileId ? Theme.palette.background.default : Theme.palette.action.selected, - borderRadius: '0rem 0rem 0.625rem 0.625rem', + borderRadius: '0rem', ':hover': { - backgroundColor: Theme.palette.background.paper, + backgroundColor: + Workspace.fileId === fileId + ? Theme.palette.background.default + : alpha(Theme.palette.background.default, 0.5), }, cursor: 'pointer', display: 'flex', diff --git a/app/front-end/src/features/editor/components/toolbarView/toolbarGroupItem.tsx b/app/front-end/src/features/editor/components/toolbarView/toolbarGroupItem.tsx index 32ed57a..cdf4858 100644 --- a/app/front-end/src/features/editor/components/toolbarView/toolbarGroupItem.tsx +++ b/app/front-end/src/features/editor/components/toolbarView/toolbarGroupItem.tsx @@ -1,5 +1,5 @@ import { SvgIconComponent } from '@mui/icons-material'; -import { Box, Button, useTheme } from '@mui/material'; +import { alpha, Box, Button, useTheme } from '@mui/material'; export interface ToolbarGroupItemProps { group: string; @@ -44,11 +44,11 @@ export const ToolbarGroupItem: React.FC = ({ icon: Icon, onClick={() => onClick()} sx={{ color: Theme.palette.text.primary, - bgcolor: Theme.palette.background.default, + backgroundColor: alpha(Theme.palette.action.selected, 0.5), borderRadius: '0.625rem', px: '1rem', '&:hover': { - bgcolor: Theme.palette.action.hover, + backgroundColor: Theme.palette.action.selected, }, }} > diff --git a/app/front-end/src/features/editor/components/toolbarView/toolbarGroupsSelectorItem.tsx b/app/front-end/src/features/editor/components/toolbarView/toolbarGroupsSelectorItem.tsx index 0865c5d..54e78be 100644 --- a/app/front-end/src/features/editor/components/toolbarView/toolbarGroupsSelectorItem.tsx +++ b/app/front-end/src/features/editor/components/toolbarView/toolbarGroupsSelectorItem.tsx @@ -1,4 +1,5 @@ import { Button, useTheme } from '@mui/material'; +import { alpha } from '@mui/system'; export interface ToolbarGroupsSelectorItemProps { id: string; @@ -42,12 +43,13 @@ export const ToolbarGroupsSelectorItem: React.FC sx={{ height: '100%', bgcolor: groupRef === id ? Theme.palette.background.paper : Theme.palette.action.selected, - borderRadius: '0.625rem 0.625rem 0rem 0rem', + borderRadius: '0rem', px: '3rem', fontWeight: 'bold', color: Theme.palette.text.primary, ':hover': { - backgroundColor: Theme.palette.background.paper, + backgroundColor: + groupRef === id ? Theme.palette.background.paper : alpha(Theme.palette.background.paper, 0.5), }, }} onClick={() => onClick()} diff --git a/app/front-end/src/stores/themeContextProvider.tsx b/app/front-end/src/stores/themeContextProvider.tsx index 637e8d7..b82b8c7 100644 --- a/app/front-end/src/stores/themeContextProvider.tsx +++ b/app/front-end/src/stores/themeContextProvider.tsx @@ -62,6 +62,7 @@ export const ThemeContextProvider: React.FC = ({ children }) => { mode: mode as PaletteMode, primary: { main: mode === 'light' ? Colors.primaryLight : Colors.primaryDark, + contrastText: mode === 'light' ? Colors.contrastTextLight : Colors.contrastTextDark, }, secondary: { main: mode === 'light' ? Colors.secondaryLight : Colors.secondaryDark, @@ -75,7 +76,6 @@ export const ThemeContextProvider: React.FC = ({ children }) => { paper: mode === 'light' ? Colors.backgroundSecondaryLight : Colors.backgroundSecondaryDark, }, action: { - hover: mode === 'light' ? Colors.secondaryLight : Colors.secondaryDark, selected: mode === 'light' ? Colors.backgroundActiveLight : Colors.backgroundActiveDark, }, error: { diff --git a/app/front-end/src/types/enums/colors.ts b/app/front-end/src/types/enums/colors.ts index dfbb119..18e8d87 100644 --- a/app/front-end/src/types/enums/colors.ts +++ b/app/front-end/src/types/enums/colors.ts @@ -6,7 +6,7 @@ * ensure consistency in styling and theming throughout the application. * * The enum includes: - * + * * **Light Mode Colors:** * - `primaryLight`: Primary color for light mode, typically used for main UI elements. * - `secondaryLight`: Secondary color for light mode, used for supporting elements. @@ -15,7 +15,7 @@ * - `backgroundPrimaryLight`: Primary background color for light mode. * - `backgroundSecondaryLight`: Secondary background color for light mode. * - `backgroundActiveLight`: Background color for active states in light mode. - * + * * **Dark Mode Colors:** * - `primaryDark`: Primary color for dark mode, typically used for main UI elements. * - `secondaryDark`: Secondary color for dark mode, used for supporting elements. @@ -24,13 +24,13 @@ * - `backgroundPrimaryDark`: Primary background color for dark mode. * - `backgroundSecondaryDark`: Secondary background color for dark mode. * - `backgroundActiveDark`: Background color for active states in dark mode. - * + * * **Global Colors:** * - `error`: Color used to indicate error states or messages. * - `success`: Color used to indicate success states or messages. * - `warning`: Color used to indicate warning states or messages. * - `info`: Color used to indicate informational messages. - * + * * @enum {string} * @property {string} primaryLight - The primary color used in light mode. * @property {string} secondaryLight - The secondary color used in light mode. @@ -54,38 +54,48 @@ * @example * // Example usage of Colors * import { Colors } from './path/to/colors'; - * + * * const headerStyle = { * backgroundColor: Colors.primaryLight, * color: Colors.textPrimaryLight, * }; - * + * * const errorMessageStyle = { * color: Colors.error, * }; */ + +/* eslint-disable @typescript-eslint/no-duplicate-enum-values */ +// Some values are duplicated for now, this way we can keep values organized and future-proof + export enum Colors { // Light mode - primaryLight = '#4C7380', + mainNavigationBackgroundLight = '#4C7380', + primaryLight = '#336e82', secondaryLight = '#D8E4E8', textPrimaryLight = '#404040', - textSecondaryLight = '#999999', + textSecondaryLight = '#525252', backgroundPrimaryLight = '#F9FBFB', backgroundSecondaryLight = '#EDEDED', backgroundActiveLight = '#CDCDCD', + contrastTextLight = '#F9FBFB', + // Dark mode - primaryDark = '#203238', - secondaryDark = '#212829', + mainNavigationBackgroundDark = '#1C2427', + primaryDark = '#6AA1AD', + secondaryDark = '#222C30', textPrimaryDark = '#F6F6F6', textSecondaryDark = '#B9B9B9', - backgroundPrimaryDark = '#252525', - backgroundSecondaryDark = '#222222', - backgroundActiveDark = '#353535', + backgroundPrimaryDark = '#292929', + backgroundSecondaryDark = '#252525', + backgroundActiveDark = '#333333', + + contrastTextDark = '#F9FBFB', // Global error = '#FF6961',