From 4fa4e4466d13df1e06796fcaa1ac6c617f47dcdb Mon Sep 17 00:00:00 2001 From: "Thomas F. K. Jorna" Date: Mon, 2 Aug 2021 23:35:35 +0200 Subject: feature: tag coloring and filtering --- components/config.ts | 5 +- components/tweaks.tsx | 228 +++++++++++++++++++++++++++++++++++++++----------- 2 files changed, 185 insertions(+), 48 deletions(-) (limited to 'components') 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) => { > - + Tag filters - - + + - Tag colors + Tag Colors - + + + @@ -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([]) + + return ( + 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([]) return ( - - - setTagFilterText(v.target.value)} - /> - { - if (!tags.includes(tagFilterText)) { - return - } - setFilter({ ...filter, tags: [...filter.tags, tagFilterText] }) - setTagFilterText('') - }} - aria-label="add tag" - icon={} - /> - - {filter.tags.map((tag) => { - return ( - - {tag} - } - onClick={() => - setFilter({ - ...filter, - tags: filter.tags.filter((t) => t !== tag), - }) - } - /> - - ) - })} - + + { + 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} + /> + } + align="stretch" + color="gray.800" + > + {Object.keys(tagColors).map((tag) => { + return ( + + + {tag} + + + + {} + + + {' '} + + {colorList.map((color: string) => ( + + setTagColors({ + ...tagColors, + [tag]: color, + }) + } + justifyContent="space-between" + alignItems="center" + display="flex" + > + + + ))} + + + + } + 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)) + }} + /> + + ) + })} + + ) } -- cgit v1.2.3