Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(query): Add a "Search" tab in the Sidebar for wildcard queries. #107

Merged
merged 27 commits into from
Nov 16, 2024
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
5094abf
refactor SearchTabPanel from new log viewer folder back to current lo…
Henry8192 Oct 24, 2024
dcada93
Rewrite the ResultsGroup return type
Henry8192 Oct 25, 2024
ac22041
show queryResults in search tab panel
Henry8192 Oct 28, 2024
908b6db
fix search results not showing; show contexts before and after search…
Henry8192 Oct 31, 2024
b931505
Nasty solution for individual search results expansion control
Henry8192 Nov 1, 2024
3ddd96a
limit max query results to 1000; add jsdoc for ResultsGroup
Henry8192 Nov 4, 2024
2cbb032
change ResultsGroup structure: each should be a page of results; rewr…
Henry8192 Nov 4, 2024
0889361
amend stylelint
Henry8192 Nov 4, 2024
d8cd28c
edit QUERY_RESULT protocol to show search progress
Henry8192 Nov 5, 2024
4709312
display search progress in status bar
Henry8192 Nov 7, 2024
a8a1551
move search progress bar under search input box
Henry8192 Nov 7, 2024
67957ca
search results now support jump to current log event
Henry8192 Nov 7, 2024
640b3e3
address issues in code review
Henry8192 Nov 11, 2024
477ded6
Rename "search" -> "query" where appropriate; fix CSS issues.
Henry8192 Nov 11, 2024
e1dc2df
Fix panel overflow when there are a few query results; change progres…
Henry8192 Nov 11, 2024
ec8cc58
clean up sx props to css files
Henry8192 Nov 12, 2024
96606b7
Apply suggestions from code review
Henry8192 Nov 12, 2024
854293f
Resolve issues from code review
Henry8192 Nov 12, 2024
30481f9
fix lint
Henry8192 Nov 12, 2024
8a81a87
Apply suggestions from code review
Henry8192 Nov 14, 2024
2c10ab7
resolve rest of suggestions
Henry8192 Nov 14, 2024
fb12080
fix lint
Henry8192 Nov 14, 2024
34a8043
Merge branch 'main' into search-tab-panel
Henry8192 Nov 14, 2024
f71269e
Update src/components/CentralContainer/Sidebar/SidebarTabs/SearchTabP…
Henry8192 Nov 15, 2024
e002f4f
Merge branch 'main' into search-tab-panel
junhaoliao Nov 15, 2024
8793d0b
Rename `DecodeResultType` -> `DecodeResult` as a result of merging fr…
junhaoliao Nov 15, 2024
580b876
Merge branch 'main' into search-tab-panel
Henry8192 Nov 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import React from "react";

import {
ButtonGroup,
DialogContent,
DialogTitle,
TabPanel,
Expand All @@ -14,6 +15,7 @@ interface CustomTabPanelProps {
children: React.ReactNode,
tabName: string,
title: string,
titleButtons?: React.ReactNode,
}

/**
Expand All @@ -23,9 +25,15 @@ interface CustomTabPanelProps {
* @param props.children
* @param props.tabName
* @param props.title
* @param props.titleButtons
* @return
*/
const CustomTabPanel = ({children, tabName, title}: CustomTabPanelProps) => {
const CustomTabPanel = ({
children,
tabName,
title,
titleButtons,
}: CustomTabPanelProps) => {
return (
<TabPanel
className={"sidebar-tab-panel"}
Expand All @@ -38,6 +46,13 @@ const CustomTabPanel = ({children, tabName, title}: CustomTabPanelProps) => {
>
{title}
</Typography>
<ButtonGroup
size={"sm"}
spacing={"1px"}
variant={"plain"}
>
{titleButtons}
</ButtonGroup>
</DialogTitle>
<DialogContent>
{children}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
.result-button {
user-select: none;

overflow: hidden;
display: flex !important;

text-overflow: ellipsis;
white-space: nowrap;
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Consider removing the !important flag from the display property.

The use of !important suggests potential specificity issues. Consider increasing selector specificity or restructuring CSS to avoid specificity wars.

-    display: flex !important;
+    display: flex;

If you need to override conflicting styles, consider using a more specific selector like:

.sidebar .search-tab-panel .result-button {
    display: flex;
}

Henry8192 marked this conversation as resolved.
Show resolved Hide resolved

.result-text {
display: inline !important;
font-size: 0.875rem !important;
font-weight: 400 !important;
}

.before-match,
.match,
.after-match {
overflow: hidden;
flex: 1 1 0; /* Allow each item to take an equal part */
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
text-overflow: ellipsis;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
import {
ListItemButton,
Typography,
} from "@mui/joy";

import "./Result.css";

Henry8192 marked this conversation as resolved.
Show resolved Hide resolved

interface ResultProps {
message: string,
matchRange: [number, number]
}
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved

/**
* Displays a button containing a message, which highlights a specific range of text.
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
*
* @param props
* @param props.message
* @param props.matchRange A two-element array indicating the start and end indices of the substring
* to be highlighted.
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
* @return
*/
const Result = ({message, matchRange}: ResultProps) => {
const [
beforeMatch,
match,
afterMatch,
] = [
message.slice(0, matchRange[0]),
message.slice(...matchRange),
message.slice(matchRange[1]),
];

Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
return (
<ListItemButton className={"result-button"}>
<Typography
className={"result-text before-match"}
level={"body-xs"}
>
{beforeMatch}
</Typography>
<Typography
className={"result-text match"}
level={"body-xs"}
sx={{
backgroundColor: "warning.softBg",
}}
>
{match}
</Typography>
<Typography
className={"result-text after-match"}
level={"body-xs"}
>
{afterMatch}
</Typography>
</ListItemButton>
);
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Enhance accessibility and error handling.

The component needs improvements in:

  1. Accessibility attributes for screen readers
  2. Keyboard navigation support
  3. Error boundaries for invalid ranges
-    <ListItemButton className={"result-button"}>
+    <ListItemButton
+        className={"result-button"}
+        role="option"
+        aria-selected={false}
+        onKeyDown={(e) => {
+            if (e.key === 'Enter' || e.key === ' ') {
+                // Handle selection
+            }
+        }}
+    >
         <Typography
             className={"result-text before-match"}
             level={"body-xs"}
+            component="span"
         >
             {beforeMatch}
         </Typography>
         <Typography
             className={"result-text match"}
             level={"body-xs"}
+            component="mark"
             sx={{
                 backgroundColor: "warning.softBg",
             }}
+            aria-label={`Matched text: ${match}`}
         >
             {match}
         </Typography>

};

Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
export default Result;
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
.results-group-title-button {
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
flex-direction: row-reverse !important;
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
gap: 2px !important;
padding-inline-start: 0 !important;
}

.results-group-title-container {
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
display: flex;
flex-grow: 1;
}

.results-group-title-text-container {
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
flex-grow: 1;
gap: 0.1rem;
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
align-items: center;
}

.results-group-content {
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
margin-left: 1.5px !important;
/* stylelint-disable-next-line custom-property-pattern */
border-left: 1px solid var(--joy-palette-neutral-outlinedBorder, #cdd7e1);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,105 @@
import {
useEffect,
useState,
} from "react";
import * as React from "react";
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved

import {
Accordion,
AccordionDetails,
AccordionSummary,
Box,
Chip,
List,
Stack,
Typography,
} from "@mui/joy";

import DescriptionOutlinedIcon from "@mui/icons-material/DescriptionOutlined";

import {QueryResultsType} from "../../../../../typings/worker";
import Result from "./Result";

import "./ResultsGroup.css";


interface ResultsGroupProps {
isAllExpanded: boolean,
pageNum: number,
results: QueryResultsType[],
}

/**
* Renders a group of results. Each group contains a list of results from a single page.
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
*
* @param props
* @param props.isAllExpanded
* @param props.pageNum
* @param props.results
* @return
*/
const ResultsGroup = React.memo(({
isAllExpanded,
pageNum,
results,
}: ResultsGroupProps) => {
const [isExpanded, setIsExpanded] = useState<boolean>(isAllExpanded);

const handleAccordionChange = (
_: React.SyntheticEvent,
newValue: boolean
) => {
setIsExpanded(newValue);
};

// On `isAllExpanded` updates, sync current results group's expand status.
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
useEffect(() => {
setIsExpanded(isAllExpanded);
}, [isAllExpanded]);
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved

return (
<Accordion
expanded={isExpanded}
onChange={handleAccordionChange}
>
<AccordionSummary
slotProps={{
button: {className: "results-group-title-button"},
}}
>
<Box className={"results-group-title-container"}>
<Stack
className={"results-group-title-text-container"}
direction={"row"}
>
<DescriptionOutlinedIcon fontSize={"inherit"}/>
<Typography
level={"title-sm"}
>
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
Page
{" "}
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
{pageNum}
</Typography>
</Stack>
<Chip size={"sm"}>
{results.length}
</Chip>
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
</Box>
</AccordionSummary>
<AccordionDetails className={"results-group-content"}>
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
<List size={"sm"}>
{results.map((r, index) => (
<Result
key={index}
matchRange={r.matchRange}
message={r.message}/>
))}
</List>
</AccordionDetails>
</Accordion>
);
});

ResultsGroup.displayName = "ResultsGroup";

Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
export default ResultsGroup;
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
import React, {
useContext,
useRef,
useState,
} from "react";

import {
AccordionGroup,
IconButton,
Textarea,
ToggleButtonGroup,
} from "@mui/joy";

import UnfoldLessIcon from "@mui/icons-material/UnfoldLess";
import UnfoldMoreIcon from "@mui/icons-material/UnfoldMore";

import {StateContext} from "../../../../../contexts/StateContextProvider";
import {
TAB_DISPLAY_NAMES,
TAB_NAME,
} from "../../../../../typings/tab";
import CustomTabPanel from "../CustomTabPanel";
import TitleButton from "../TitleButton";
import ResultsGroup from "./ResultsGroup";


enum SEARCH_OPTION {
IS_CASE_SENSITIVE = "isCaseSensitive",
IS_REGEX = "isRegex"
}

/**
* Displays a panel for submitting search queries and viewing query results.
*
* @return
*/
const SearchTabPanel = () => {
const [isAllExpanded, setIsAllExpanded] = useState<boolean>(true);
const [searchOptions, setSearchOptions] = useState<SEARCH_OPTION[]>([]);
const searchTextRef = useRef<HTMLTextAreaElement>(null);
const {queryResults, startQuery} = useContext(StateContext);
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
const handleSearch = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
if ("Enter" === event.key && searchTextRef.current) {
event.preventDefault();
const isCaseSensitive = searchOptions.includes(SEARCH_OPTION.IS_CASE_SENSITIVE);
const isRegex = searchOptions.includes(SEARCH_OPTION.IS_REGEX);
startQuery(searchTextRef.current.value, isRegex, isCaseSensitive);
}
};
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Add performance optimizations and error handling.

Given the reported issues with browser freezing on large result sets, consider implementing:

  1. Debounce the search operation to prevent excessive queries
  2. Add loading state to provide user feedback
  3. Implement error handling for failed searches

Here's a suggested implementation:

+ import { useState, useRef, useContext, useCallback } from 'react';
+ import debounce from 'lodash/debounce';

  const SearchTabPanel = () => {
    const [isAllExpanded, setIsAllExpanded] = useState<boolean>(true);
    const [searchOptions, setSearchOptions] = useState<SEARCH_OPTION[]>([]);
+   const [isLoading, setIsLoading] = useState<boolean>(false);
+   const [error, setError] = useState<string | null>(null);
    const searchTextRef = useRef<HTMLTextAreaElement>(null);
    const {queryResults, startQuery} = useContext(StateContext);

-   const handleSearch = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {
+   const executeSearch = useCallback(async (searchText: string) => {
+     try {
+       setIsLoading(true);
+       setError(null);
        const isCaseSensitive = searchOptions.includes(SEARCH_OPTION.IS_CASE_SENSITIVE);
        const isRegex = searchOptions.includes(SEARCH_OPTION.IS_REGEX);
-       startQuery(searchTextRef.current.value, isRegex, isCaseSensitive);
+       await startQuery(searchText, isRegex, isCaseSensitive);
+     } catch (err) {
+       setError(err instanceof Error ? err.message : 'Search failed');
+     } finally {
+       setIsLoading(false);
+     }
+   }, [searchOptions, startQuery]);

+   const debouncedSearch = useCallback(
+     debounce((searchText: string) => executeSearch(searchText), 300),
+     [executeSearch]
+   );

+   const handleSearch = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {
      if ("Enter" === event.key && searchTextRef.current) {
        event.preventDefault();
+       debouncedSearch(searchTextRef.current.value);
      }
    };
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
const [isAllExpanded, setIsAllExpanded] = useState<boolean>(true);
const [searchOptions, setSearchOptions] = useState<SEARCH_OPTION[]>([]);
const searchTextRef = useRef<HTMLTextAreaElement>(null);
const {queryResults, startQuery} = useContext(StateContext);
const handleSearch = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {
if ("Enter" === event.key && searchTextRef.current) {
event.preventDefault();
const isCaseSensitive = searchOptions.includes(SEARCH_OPTION.IS_CASE_SENSITIVE);
const isRegex = searchOptions.includes(SEARCH_OPTION.IS_REGEX);
startQuery(searchTextRef.current.value, isRegex, isCaseSensitive);
}
};
const [isAllExpanded, setIsAllExpanded] = useState<boolean>(true);
const [searchOptions, setSearchOptions] = useState<SEARCH_OPTION[]>([]);
const [isLoading, setIsLoading] = useState<boolean>(false);
const [error, setError] = useState<string | null>(null);
const searchTextRef = useRef<HTMLTextAreaElement>(null);
const {queryResults, startQuery} = useContext(StateContext);
const executeSearch = useCallback(async (searchText: string) => {
try {
setIsLoading(true);
setError(null);
const isCaseSensitive = searchOptions.includes(SEARCH_OPTION.IS_CASE_SENSITIVE);
const isRegex = searchOptions.includes(SEARCH_OPTION.IS_REGEX);
await startQuery(searchText, isRegex, isCaseSensitive);
} catch (err) {
setError(err instanceof Error ? err.message : 'Search failed');
} finally {
setIsLoading(false);
}
}, [searchOptions, startQuery]);
const debouncedSearch = useCallback(
debounce((searchText: string) => executeSearch(searchText), 300),
[executeSearch]
);
const handleSearch = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {
if ("Enter" === event.key && searchTextRef.current) {
event.preventDefault();
debouncedSearch(searchTextRef.current.value);
}
};

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Enhance search handler with debouncing and error handling.

The current implementation may cause performance issues with large result sets. Consider implementing:

  1. Debounce the search operation
  2. Add loading state
  3. Implement error handling
  4. Validate regex patterns
+import { debounce } from 'lodash';
+
 const SearchTabPanel = () => {
+    const [isLoading, setIsLoading] = useState(false);
+    const [error, setError] = useState<string | null>(null);
+
+    const executeSearch = useCallback(async (
+        searchText: string,
+        isRegex: boolean,
+        isCaseSensitive: boolean
+    ) => {
+        try {
+            if (isRegex) {
+                new RegExp(searchText); // Validate regex
+            }
+            setIsLoading(true);
+            setError(null);
+            await startQuery(searchText, isRegex, isCaseSensitive);
+        } catch (err) {
+            setError(err instanceof Error ? err.message : 'Search failed');
+        } finally {
+            setIsLoading(false);
+        }
+    }, [startQuery]);
+
+    const debouncedSearch = useMemo(
+        () => debounce(executeSearch, 300),
+        [executeSearch]
+    );
+
     const handleSearch = (event: React.KeyboardEvent<HTMLTextAreaElement>) => {
         if ("Enter" === event.key && searchTextRef.current) {
             event.preventDefault();
             const isCaseSensitive = searchOptions.includes(SEARCH_OPTION.IS_CASE_SENSITIVE);
             const isRegex = searchOptions.includes(SEARCH_OPTION.IS_REGEX);
-            startQuery(searchTextRef.current.value, isRegex, isCaseSensitive);
+            debouncedSearch(searchTextRef.current.value, isRegex, isCaseSensitive);
         }
     };

Committable suggestion skipped: line range outside the PR's diff.


return (
<CustomTabPanel
tabName={TAB_NAME.SEARCH}
title={TAB_DISPLAY_NAMES[TAB_NAME.SEARCH]}
titleButtons={
<TitleButton onClick={() => { setIsAllExpanded((v) => !v); }}>
{isAllExpanded ?
<UnfoldLessIcon/> :
<UnfoldMoreIcon/>}
</TitleButton>
}
>
<Textarea
maxRows={7}
placeholder={"Search"}
size={"sm"}
sx={{flexDirection: "row", zIndex: 0}}
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
endDecorator={
<ToggleButtonGroup
size={"sm"}
spacing={0.3}
sx={{borderRadius: "2px"}}
value={searchOptions}
variant={"plain"}
onChange={(_, newValue) => {
setSearchOptions(newValue);
}}
>
<IconButton
sx={{fontFamily: "Inter"}}
value={SEARCH_OPTION.IS_CASE_SENSITIVE}
>
Aa
</IconButton>
<IconButton
sx={{fontFamily: "Inter"}}
value={SEARCH_OPTION.IS_REGEX}
>
.*
</IconButton>
</ToggleButtonGroup>
}
slotProps={{
textarea: {ref: searchTextRef},
endDecorator: {sx: {marginBlockStart: 0, display: "block"}},
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
}}
onKeyDown={handleSearch}/>
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
<AccordionGroup
disableDivider={true}
size={"sm"}
>
{Array.from(queryResults.entries()).map(([pageNum, results]) => (
<ResultsGroup
isAllExpanded={isAllExpanded}
key={`${pageNum} + ${results.length}`}
pageNum={pageNum}
results={results}/>
))}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Improve key generation and implement virtualization.

The current key generation using array length could cause unnecessary re-renders. Additionally, for large result sets, implement virtualization:

-{Array.from(queryResults.entries()).map(([pageNum, results]) => (
-    <ResultsGroup
-        isAllExpanded={isAllExpanded}
-        key={`${pageNum} + ${results.length}`}
-        pageNum={pageNum}
-        results={results}/>
-))}
+<VirtualizedList
+    height={400}
+    itemCount={queryResults.size}
+    itemSize={50}
+    width="100%"
+>
+    {({ index, style }) => {
+        const [pageNum, results] = Array.from(queryResults.entries())[index];
+        return (
+            <div style={style}>
+                <ResultsGroup
+                    isAllExpanded={isAllExpanded}
+                    key={pageNum}
+                    pageNum={pageNum}
+                    results={results}
+                />
+            </div>
+        );
+    }}
+</VirtualizedList>

Committable suggestion skipped: line range outside the PR's diff.

</AccordionGroup>
</CustomTabPanel>
);
};

Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
export default SearchTabPanel;
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
.sidebar-tab-title-button {
z-index: 0 !important;
Henry8192 marked this conversation as resolved.
Show resolved Hide resolved
min-width: 0 !important;
min-height: 0 !important;
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Consider alternatives to !important declarations.

Using !important suggests specificity conflicts with MUI's styles. Consider these alternatives:

  1. Use more specific selectors
  2. Customize MUI's theme to override default styles
  3. Use MUI's sx prop for component-specific styles

Example of using a more specific selector:

-.sidebar-tab-title-button {
+.sidebar-tabs .MuiIconButton-root.sidebar-tab-title-button {
     z-index: 0;
     min-width: 0;
     min-height: 0;
 }
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
.sidebar-tab-title-button {
z-index: 0 !important;
min-width: 0 !important;
min-height: 0 !important;
}
.sidebar-tabs .MuiIconButton-root.sidebar-tab-title-button {
z-index: 0;
min-width: 0;
min-height: 0;
}

Loading
Loading