diff options
author | Thomas F. K. Jorna <[email protected]> | 2021-08-02 23:35:35 +0200 |
---|---|---|
committer | Thomas F. K. Jorna <[email protected]> | 2021-08-02 23:35:35 +0200 |
commit | 4fa4e4466d13df1e06796fcaa1ac6c617f47dcdb (patch) | |
tree | cb1ad05b966dd1f0c7e3d4f84ae5a5708cb33e97 /components | |
parent | c2b219c06a6bf37d35f6eac7879e04d04bd5d023 (diff) |
feature: tag coloring and filtering
Diffstat (limited to 'components')
-rw-r--r-- | components/config.ts | 5 | ||||
-rw-r--r-- | components/tweaks.tsx | 228 |
2 files changed, 185 insertions, 48 deletions
diff --git a/components/config.ts b/components/config.ts index e9471f8..0544b58 100644 --- a/components/config.ts +++ b/components/config.ts @@ -35,7 +35,6 @@ export const initialFilter = { orphans: false, parents: true, tags: [], - tagColors: [], nodes: [], links: [], date: [], @@ -89,6 +88,10 @@ export const initialVisuals = { citeNodeColor: 'black', } +export interface TagColors { + [tag: string]: string +} + export const initialBehavior = { follow: 'zoom', localSame: 'add', diff --git a/components/tweaks.tsx b/components/tweaks.tsx index e46baf3..cb1fec0 100644 --- a/components/tweaks.tsx +++ b/components/tweaks.tsx @@ -8,6 +8,7 @@ import { ArrowRightIcon, AddIcon, DeleteIcon, + CheckCircleIcon, } from '@chakra-ui/icons' import { Accordion, @@ -42,6 +43,8 @@ import { SlideFade, Input, } from '@chakra-ui/react' +import { CUIAutoComplete } from 'chakra-ui-autocomplete' + import React, { useState, useContext } from 'react' import Scrollbars from 'react-custom-scrollbars-2' import { @@ -50,6 +53,7 @@ import { initialVisuals, initialMouse, initialBehavior, + TagColors, } from './config' import { ThemeContext } from '../util/themecontext' @@ -68,6 +72,8 @@ export interface TweakProps { behavior: typeof initialBehavior setBehavior: any tags: string[] + tagColors: TagColors + setTagColors: any } export const Tweaks = (props: TweakProps) => { @@ -85,6 +91,8 @@ export const Tweaks = (props: TweakProps) => { behavior, setBehavior, tags, + tagColors, + setTagColors, } = props const [showTweaks, setShowTweaks] = useState(true) const { highlightColor, setHighlightColor } = useContext(ThemeContext) @@ -97,9 +105,6 @@ export const Tweaks = (props: TweakProps) => { 'blue.500', 'pink.500', 'purple.500', - 'gray.400', - 'gray.500', - 'gray.600', 'white', 'gray.100', 'gray.200', @@ -229,22 +234,35 @@ export const Tweaks = (props: TweakProps) => { ></Switch> </Flex> </VStack> - <Accordion allowToggle allowMultiple paddingLeft={3}> + <Accordion padding={0} allowToggle allowMultiple paddingLeft={3}> <AccordionItem> <AccordionButton> Tag filters <AccordionIcon /> </AccordionButton> - <AccordionPanel> - <TagPanel filter={filter} setFilter={setFilter} tags={tags} /> + <AccordionPanel pr={0} mr={0}> + <TagPanel + highlightColor={highlightColor} + filter={filter} + setFilter={setFilter} + tags={tags} + /> </AccordionPanel> </AccordionItem> <AccordionItem> <AccordionButton> - Tag colors + Tag Colors <AccordionIcon /> </AccordionButton> - <AccordionPanel></AccordionPanel> + <AccordionPanel pr={0} mr={0}> + <TagColorPanel + tags={tags} + colorList={colorList} + tagColors={tagColors} + setTagColors={setTagColors} + highlightColor={highlightColor} + /> + </AccordionPanel> </AccordionItem> </Accordion> </AccordionPanel> @@ -1271,49 +1289,165 @@ export interface TagPanelProps { tags: string[] filter: typeof initialFilter setFilter: any + highlightColor: string } export const TagPanel = (props: TagPanelProps) => { - const { filter, setFilter, tags } = props - const [tagFilterText, setTagFilterText] = useState('') + const { filter, setFilter, tags, highlightColor } = props + const tagArray = tags.map((tag) => { + return { value: tag, label: tag } + }) + // .concat[{ value: 'placeholder', label: 'New filter' }] + + const [selectedItems, setSelectedItems] = useState<typeof tagArray>([]) + + return ( + <CUIAutoComplete + items={tagArray} + label="Add tag to filter" + placeholder=" " + onCreateItem={(item) => null} + disableCreateItem={true} + selectedItems={selectedItems} + onSelectedItemsChange={(changes) => { + if (changes.selectedItems) { + setSelectedItems(changes.selectedItems) + setFilter({ ...filter, tags: selectedItems.map((item) => item.value) }) + } + }} + listItemStyleProps={{ overflow: 'hidden' }} + highlightItemBg="gray.400" + toggleButtonStyleProps={{ variant: 'outline' }} + inputStyleProps={{ + focusBorderColor: highlightColor, + color: 'gray.800', + borderColor: 'gray.600', + }} + tagStyleProps={{ + rounded: 'full', + bg: highlightColor, + height: 8, + paddingLeft: 4, + fontWeight: 'bold', + }} + hideToggleButton + itemRenderer={(selected) => selected.label} + /> + ) +} + +export interface TagColorPanelProps { + tags: string[] + highlightColor: string + colorList: string[] + tagColors: TagColors + setTagColors: any +} +export const TagColorPanel = (props: TagColorPanelProps) => { + const { colorList, tagColors, setTagColors, highlightColor, tags } = props + const tagArray = tags.map((tag) => { + return { value: tag, label: tag } + }) + + const [selectedItems, setSelectedItems] = useState<typeof tagArray>([]) return ( - <VStack> - <Flex alignItems="center" justifyContent="space-between"> - <Input - placeHolder="New tag..." - value={tagFilterText} - onChange={(v) => setTagFilterText(v.target.value)} - /> - <IconButton - onClick={() => { - if (!tags.includes(tagFilterText)) { - return - } - setFilter({ ...filter, tags: [...filter.tags, tagFilterText] }) - setTagFilterText('') - }} - aria-label="add tag" - icon={<AddIcon />} - /> - </Flex> - {filter.tags.map((tag) => { - return ( - <Flex aignItems="center" justifyContent="space-between"> - {tag} - <IconButton - aria-label={'delete tag ' + tag} - icon={<DeleteIcon />} - onClick={() => - setFilter({ - ...filter, - tags: filter.tags.filter((t) => t !== tag), - }) - } - /> - </Flex> - ) - })} - </VStack> + <Box> + <CUIAutoComplete + items={tagArray} + label="Add tag to filter" + placeholder=" " + disableCreateItem={true} + selectedItems={selectedItems} + onSelectedItemsChange={(changes) => { + if (changes.selectedItems) { + setSelectedItems(Array.from(new Set(changes.selectedItems))) + setTagColors( + Object.fromEntries( + Array.from(new Set(changes.selectedItems)).map((item) => { + return [item.label, tagColors[item.label] ?? 'gray.600'] + }), + ), + ) + } + }} + listItemStyleProps={{ overflow: 'hidden' }} + highlightItemBg="gray.400" + toggleButtonStyleProps={{ variant: 'outline' }} + inputStyleProps={{ + focusBorderColor: highlightColor, + color: 'gray.800', + borderColor: 'gray.600', + }} + tagStyleProps={{ + display: 'none', + rounded: 'full', + bg: highlightColor, + height: 8, + paddingLeft: 4, + fontWeight: 'bold', + }} + hideToggleButton + itemRenderer={(selected) => selected.label} + /> + <VStack + spacing={2} + justifyContent="flex-start" + divider={<StackDivider borderColor="gray.500" />} + align="stretch" + color="gray.800" + > + {Object.keys(tagColors).map((tag) => { + return ( + <Flex key={tag} alignItems="center" justifyContent="space-between" width="100%" pl={2}> + <Box width="100%"> + <Text fontWeight="bold">{tag}</Text> + </Box> + <Menu isLazy placement="right"> + <MenuButton as={Button} colorScheme="" color="black"> + {<Box bgColor={tagColors[tag]} borderRadius="sm" height={6} width={6}></Box>} + </MenuButton> + <Portal> + {' '} + <MenuList minW={10} zIndex="popover" bgColor="gray.200"> + {colorList.map((color: string) => ( + <MenuItem + key={color} + onClick={() => + setTagColors({ + ...tagColors, + [tag]: color, + }) + } + justifyContent="space-between" + alignItems="center" + display="flex" + > + <Box bgColor={color} borderRadius="sm" height={6} width={6}></Box> + </MenuItem> + ))} + </MenuList> + </Portal> + </Menu> + <IconButton + aria-label="Delete tag color" + variant="ghost" + icon={<DeleteIcon />} + onClick={() => { + setTagColors( + Object.fromEntries( + Array.from(new Set(selectedItems)).map((item) => { + return [item.label, tagColors[item.label] ?? 'gray.600'] + }), + ), + ) + setSelectedItems(selectedItems.filter((item) => item.value !== tag)) + }} + /> + </Flex> + ) + })} + </VStack> + </Box> ) } |