Skip to content

Commit

Permalink
refactor: blocks page and search (#540)
Browse files Browse the repository at this point in the history
  • Loading branch information
arthurgeron authored Sep 20, 2024
1 parent e49b834 commit a5e81a7
Show file tree
Hide file tree
Showing 28 changed files with 6,949 additions and 1,412 deletions.
3 changes: 2 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -89,4 +89,5 @@ act-actions

# playwright report
**/**/playwright-report/*
**/**/playwright-html/*
**/**/playwright-html/*
*storybook.log
2 changes: 1 addition & 1 deletion packages/app-explorer/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@
"autoprefixer": "10.4.17",
"postcss": "8.4.35",
"postcss-import": "16.0.1",
"storybook": "^8.0.8",
"storybook": "8.3.0",
"storybook-addon-theme": "workspace:*",
"tailwindcss": "3.4.4",
"tailwindcss-animate": "1.0.7",
Expand Down
6 changes: 4 additions & 2 deletions packages/app-explorer/src/app/block/[id]/layout.tsx
Original file line number Diff line number Diff line change
@@ -1,17 +1,19 @@
import type { ReactNode } from 'react';
import { getBlock } from '~/systems/Block/actions/get-block';
import { BlockHeader } from '~/systems/Block/components/BlockHeader';
import type { BlockRouteParams } from '~/systems/Block/types';

export default function BlockLayout({
export default async function BlockLayout({
children,
params: { id },
}: {
children: ReactNode;
params: BlockRouteParams;
}) {
const { producer } = await getBlock({ id });
return (
<>
<BlockHeader id={id} />
<BlockHeader producer={producer} />
{children}
</>
);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export async function Block({
}: BlockScreenProps) {
return (
<>
<BlockHeader isLoading={isLoading} id={id} />
<BlockHeader isLoading={isLoading} producer={id} />
{viewMode === ViewModes.Simple && (
<BlockScreenSimple
block={block}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,10 @@ import type { ViewModes } from '~/systems/Core/components/ViewMode/constants';
import { isValidAddress } from '~/systems/Core/utils/address';

export function BlockHeader({
id,
producer,
isLoading,
}: {
id: string;
producer: string | null | undefined;
isLoading?: boolean;
}) {
const { mode } = useParams<{ mode: ViewModes }>();
Expand All @@ -23,10 +23,11 @@ export function BlockHeader({
isLoading={isLoading}
loadingEl={<LoadingBox className="w-20 h-5 mt-1" />}
regularEl={
isValidAddress(id) ? (
<Address full value={id || ''} fixed="b256" />
//
isValidAddress(producer) ? (
<Address full value={producer || ''} fixed="b256" />
) : (
<>#{id}</>
<>#{producer}</>
)
}
/>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,14 @@ export function BlockScreenSimple({
}: BlockScreenSimpleProps) {
return (
<VStack>
<Grid className="grid-rows-3 tablet:grid-rows-1 tablet:grid-cols-3 gap-6 mb-8">
<Grid className="grid-rows-3 tablet:grid-rows-2 tablet:grid-cols-2 desktop:grid-cols-4 gap-6 mb-8">
<CardInfo name="Height" className="flex-1">
<LoadingWrapper
isLoading={isLoading}
loadingEl={<LoadingBox className="w-12 h-6" />}
regularEl={block?.height}
/>
</CardInfo>
<CardInfo name="Producer" className="flex-1">
<Address
value={producer || ''}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,13 +8,13 @@ import { styles } from './styles';
type SearchFormProps = {
className: string;
autoFocus?: boolean;
expandOnFocus?: boolean;
variablePosition?: boolean;
};

export function SearchForm({
className,
autoFocus,
expandOnFocus,
variablePosition,
}: SearchFormProps) {
const classes = styles();
const [results, action] = useFormState(
Expand All @@ -27,10 +27,10 @@ export function SearchForm({
return (
<form action={action} className={classes.searchSize()}>
<SearchInput
variablePosition={variablePosition}
className={className}
searchResult={results}
autoFocus={autoFocus}
expandOnFocus={expandOnFocus}
/>
</form>
);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,178 @@
'use client';

import type { GQLSearchResult, Maybe } from '@fuel-explorer/graphql';
import type { BaseProps, InputProps } from '@fuels/ui';
import { Focus, Icon, IconButton, Input, Tooltip, VStack } from '@fuels/ui';
import { IconCheck, IconSearch, IconX } from '@tabler/icons-react';
import type { KeyboardEvent } from 'react';
import { useContext, useRef, useState } from 'react';
import { useFormStatus } from 'react-dom';

import { cx } from '../../../utils/cx';

import { SearchResultDropdown } from '../SearchResultDropdown';
import { SearchContext } from '../SearchWidget';
import { usePropagateInputMouseClick } from '../hooks/usePropagateInputMouseClick';
import { DEFAULT_SEARCH_INPUT_WIDTH } from './constants';
import { styles } from './styles';

type SearchInputProps = BaseProps<InputProps> & {
onSubmit?: (value: string) => void;
searchResult?: Maybe<GQLSearchResult>;
alwaysDisplayActionButtons?: boolean;
variablePosition?: boolean;
};

export function SearchInput({
value: initialValue = '',
className,
autoFocus,
placeholder = 'Search here...',
searchResult,
variablePosition,
...props
}: SearchInputProps) {
const classes = styles();
const [value, setValue] = useState<string>(initialValue as string);
const [isFocused, setIsFocused] = useState(false);
const [isOpen, setIsOpen] = useState(false);
const [hasSubmitted, setHasSubmitted] = useState(false);
const inputRef = useRef<HTMLInputElement>(null);
const containerRef = useRef<HTMLDivElement>(null);
const { pending } = useFormStatus();
const { dropdownRef } = useContext(SearchContext);
const openDropdown = hasSubmitted
? !pending
: isOpen && !pending && !!searchResult;

usePropagateInputMouseClick({
containerRef,
inputRef,
enabled: openDropdown,
});

function handleChange(event: React.ChangeEvent<HTMLInputElement>) {
setValue(event.target.value);
}

function handleSubmit() {
setIsOpen(true);
setHasSubmitted(true);
}

function close() {
setIsOpen(false);
setHasSubmitted(false);
}

function handleClear() {
setValue('');
close();
}

function handleFocus() {
setIsFocused(true);
}

function handleBlur() {
setIsFocused(false);
}

function onKeyDown(e: KeyboardEvent<HTMLInputElement>) {
if (e.key === 'Enter') {
e.preventDefault();
(e.target as HTMLFormElement).form?.dispatchEvent(
new Event('submit', { cancelable: true, bubbles: true }),
);
handleSubmit();
}
}

return (
<div className="relative">
<VStack
gap="0"
data-variable-position={variablePosition}
className={classes.searchBox()}
data-active={isFocused || openDropdown}
>
<Focus.ArrowNavigator autoFocus={autoFocus}>
<div ref={containerRef} className={classes.inputContainer()}>
<Input
{...props}
ref={inputRef}
name="query"
placeholder={placeholder}
value={value}
onChange={handleChange}
variant="surface"
radius="large"
size="3"
data-active={isFocused || openDropdown}
className={cx(className, classes.inputWrapper())}
type="search"
onFocus={handleFocus}
onBlur={handleBlur}
onKeyDown={onKeyDown}
>
<div
data-show={isFocused}
className={classes.inputActionsContainer()}
>
<Input.Slot side="right">
<Tooltip content="Submit">
<IconButton
type="submit"
aria-label="Submit"
icon={IconCheck}
iconColor="text-brand"
variant="link"
className="!ml-0 tablet:ml-2"
isLoading={pending}
onClick={handleSubmit}
/>
</Tooltip>
<IconButton
aria-label="Clear"
icon={IconX}
iconColor="text-gray-11"
variant="link"
className="m-0"
onClick={handleClear}
/>
</Input.Slot>
</div>

<Input.Slot
data-show={!isFocused}
side="right"
className="[&[data-show=false]]:hidden"
>
<Icon icon={IconSearch} size={16} />
</Input.Slot>
</Input>
</div>
</Focus.ArrowNavigator>
<SearchResultDropdown
ref={dropdownRef}
width={
containerRef.current?.offsetWidth || DEFAULT_SEARCH_INPUT_WIDTH
}
searchResult={searchResult}
searchValue={value}
openDropdown={openDropdown}
isFocused={isFocused}
onSelectItem={() => {
handleClear();
}}
onOpenChange={(open) => {
setIsOpen(open);
if (!open) {
close();
}
}}
/>
</VStack>
</div>
);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export const DEFAULT_SEARCH_INPUT_WIDTH = 400;
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export * from './SearchInput';
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
import { tv } from 'tailwind-variants';

export const styles = tv({
slots: {
searchBox: [
'transition-all duration-200 [&[data-active=false]]:ease-in [&[data-active=true]]:ease-out',
'group justify-center items-center',
'block left-0 w-full', // needed for properly execution of transitions
'[&[data-variable-position=true]]:[&[data-active=true]]:w-[calc(100vw+1px)] [&[data-variable-position=true]]:[&[data-active=true]]:left-[-64px] tablet:[&[data-active=true]]:w-full',
'[&[data-active=true]]:absolute tablet:[&[data-active=true]]:left-0 [&[data-active=true]]:right-0',
'[&[data-variable-position=true]]:[&[data-active=true]]:top-[-14px] [&[data-variable-position=true]]:tablet:[&[data-active=true]]:top-[-4px] [&[data-variable-position=true]]:desktop:[&[data-active=true]]:top-[-20px]',
'[&[data-active=true]]:z-50',
],
inputContainer: 'w-full',
inputWrapper: [
'outline-none h-[40px] group-[&[data-active=true]]:h-[60px] tablet:group-[&[data-active=true]]:h-[40px]',
'group-[&[data-active=true]]:rounded-none tablet:group-[&[data-active=true]]:rounded-[var(--text-field-border-radius)] ',
'border-x-[1px] border-y-[1px] group-[&[data-active=true]]:border-x-0 group-[&[data-active=true]]:border-y-0 tablet:group-[&[data-active=true]]:border-x-[1px] tablet:group-[&[data-active=true]]:border-y-[1px]',
'border-[var(--color-border)] shadow-none',
'bg-none dark:bg-[var(--color-surface)] group-[&[data-active=true]]:bg-[var(--color-panel-solid)]',
'[&_.rt-TextFieldChrome]:bg-gray-1 [&_.rt-TextFieldChrome]:outline-none',
'[&_.rt-TextFieldChrome]:[&[data-opened=true]]:rounded-b-none',
'group-[&[data-active=true]]:[&_.rt-TextFieldChrome]:shadow-none',
],
inputActionsContainer:
'[&[data-show=false]]:hidden absolute flex items-center h-full right-0 top-0 transform',
},
});
Loading

0 comments on commit a5e81a7

Please sign in to comment.