Skip to content

Commit

Permalink
feat: refactor the code for the search component and adapt the style …
Browse files Browse the repository at this point in the history
…to the latest design draft (#251)

* refactor: move components\Search\Filter to components\Filter

* refactor: Search component

* feat: refactor the code for the search component and adapt the style to the latest design draft
  • Loading branch information
WhiteMinds authored Feb 8, 2024
1 parent f142459 commit 7b278f0
Show file tree
Hide file tree
Showing 17 changed files with 274 additions and 264 deletions.
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { useState, useRef, useEffect } from 'react'
import { useTranslation } from 'react-i18next'
import SearchBlack from '../../../assets/search_black.png'
import ClearLogo from '../../../assets/clear.png'
import SearchBlack from '../../assets/search_black.png'
import ClearLogo from '../../assets/clear.png'
import { FilterImage, FilterPanel, ResetButtonPanel, FilterInputPanel } from './styled'

const Filter = ({
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import styled from 'styled-components'
import SimpleButton from '../../SimpleButton'
import variables from '../../../styles/variables.module.scss'
import SimpleButton from '../SimpleButton'
import variables from '../../styles/variables.module.scss'

export const FilterPanel = styled.div`
width: 600px;
Expand Down
6 changes: 3 additions & 3 deletions src/components/Search/Search.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -21,12 +21,12 @@ beforeEach(() => {

test('show clear button when content is available', async () => {
initI18n()
const { getByRole, getAllByRole } = render(
const { getByRole, queryByTitle } = render(
<QueryClientProvider client={new QueryClient()}>
<Search />
</QueryClientProvider>,
)
const getClearButton = () => getAllByRole('button')[2]
const getClearButton = () => queryByTitle('clear')

const inputEl = getByRole('textbox') as HTMLInputElement
expect(inputEl).toBeInstanceOf(HTMLInputElement)
Expand All @@ -42,7 +42,7 @@ test('show clear button when content is available', async () => {
fireEvent.change(inputEl, { target: { value: 'test' } })
const btn = getClearButton()
expect(btn).toBeTruthy()
fireEvent.click(btn)
fireEvent.click(btn!)
expect(inputEl.value).toBe('')
expect(getClearButton()).toBeFalsy()
})
53 changes: 31 additions & 22 deletions src/components/Search/SearchByNameResults.module.scss
Original file line number Diff line number Diff line change
@@ -1,47 +1,56 @@
.searchResultsPanelWrapper {
position: absolute;
width: 100%;
z-index: 2;
top: calc(100% + 8px);
left: 0;
display: flex;
flex-direction: column;
gap: 12px;
width: 100%;
max-height: 292px;
padding: 12px 8px;
max-height: 232px;
overflow-y: auto;
background: white;
color: black;
top: 38px;
left: 0;
border-radius: 0 0 4px 4px;
border-radius: 4px;
box-shadow: 0 4px 4px 0 rgb(16 16 16 / 5%);
}

.loadingWrapper {
margin: 0;
}
.empty {
padding: 28px 0;
text-align: center;
font-size: 16px;
color: #333;
}

.searchResult {
user-select: none;
cursor: pointer;
display: flex;
align-items: center;
display: block;
width: 100%;
height: 44px;
border-radius: 4px;
padding: 12px 4px;
font-weight: 500;
font-size: 16px;
border-bottom: solid 1px #e5e5e5;
padding: 0 4px;
cursor: pointer;

&:hover {
background: #f5f5f5;
}
}

.content {
display: flex;
align-items: center;
gap: 8px;
width: 100%;
height: 44px;
padding: 12px 0;
border-bottom: solid 1px #e5e5e5;
}

.tokenSymbol {
min-width: 100px;
font-size: 16px;
color: #333;
}

.typeHash {
font-size: 14px;
font-weight: 500;
margin-left: 8px;
flex: 1;
min-width: 0;
color: #666;
}
69 changes: 37 additions & 32 deletions src/components/Search/SearchByNameResults.tsx
Original file line number Diff line number Diff line change
@@ -1,57 +1,62 @@
import { useTranslation } from 'react-i18next'
import classNames from 'classnames'
import { FC } from 'react'
import { UDTQueryResult } from '../../services/ExplorerService/fetcher'
import styles from './SearchByNameResults.module.scss'
import EllipsisMiddle from '../EllipsisMiddle'
import SmallLoading from '../Loading/SmallLoading'

type Props = {
loading?: boolean
udtQueryResults: UDTQueryResult[] | null
udtQueryResults: UDTQueryResult[]
}

export const SearchByNameResults = (props: Props) => {
const { udtQueryResults, loading } = props
if (loading) {
return (
<div className={styles.searchResultsPanelWrapper}>
<SmallLoading className={styles.loadingWrapper} />
</div>
)
}
if (udtQueryResults) {
return (
<div className={styles.searchResultsPanelWrapper}>
{udtQueryResults.length === 0 ? (
<EmptySearchByNameResult />
) : (
udtQueryResults.map(item => {
return <SearchByNameResult key={item.typeHash} item={item} />
})
)}
</div>
)
}
return null
}

const EmptySearchByNameResult = () => {
export const SearchByNameResults: FC<Props> = ({ udtQueryResults, loading }) => {
const { t } = useTranslation()
return <>{t('search.no_search_result')}</>

return (
<div className={styles.searchResultsPanelWrapper}>
{/* eslint-disable-next-line no-nested-ternary */}
{loading ? (
<SmallLoading className={styles.loadingWrapper} />
) : udtQueryResults.length === 0 ? (
<div className={styles.empty}>{t('search.no_search_result')}</div>
) : (
udtQueryResults.map(item => <SearchByNameResult key={item.typeHash} item={item} />)
)}
</div>
)
}

const SearchByNameResult = (props: { item: UDTQueryResult }) => {
const SearchByNameResult: FC<{ item: UDTQueryResult }> = ({ item }) => {
const { t } = useTranslation()
const { item } = props
const { typeHash, fullName, symbol, udtType } = item
const displayName = symbol ?? fullName

return (
<a
className={styles.searchResult}
href={`${window.origin}/${udtType === 'omiga_inscription' ? 'inscription' : 'sudt'}/${typeHash}`}
>
<EllipsisMiddle className={styles.tokenSymbol}>{displayName ?? t('udt.unknown_token')}</EllipsisMiddle>
<EllipsisMiddle className={classNames(styles.typeHash, 'monospace')}>{typeHash}</EllipsisMiddle>
<div className={styles.content}>
{/* TODO: Need to implement highlighting for the matched keywords. */}
<EllipsisMiddle
className={styles.tokenSymbol}
style={{ maxWidth: 'min(180px, 40%)' }}
useTextWidthForPlaceholderWidth
title={displayName}
>
{displayName ?? t('udt.unknown_token')}
</EllipsisMiddle>
<EllipsisMiddle
className={classNames(styles.typeHash, 'monospace')}
style={{ maxWidth: 'min(200px, 60%)' }}
useTextWidthForPlaceholderWidth
title={typeHash}
>
{typeHash}
</EllipsisMiddle>
</div>
</a>
)
}
3 changes: 3 additions & 0 deletions src/components/Search/clear.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
50 changes: 29 additions & 21 deletions src/components/Search/index.module.scss
Original file line number Diff line number Diff line change
@@ -1,30 +1,11 @@
@import '../../styles/variables.module';

.byNameOrId {
// remove original button styles
border: none;
display: flex;
padding: 2px 8px;
justify-content: center;
align-items: center;
margin-right: 16px;
color: #666;
font-size: 14px;
opacity: 0.8;
cursor: pointer;
user-select: none;
border-radius: 2px;
background: #f0f0f0;
white-space: nowrap;
}

.searchInputPanel {
position: relative;
width: 100%;
height: 100%;
font-size: 14px;
padding-left: 10px;
padding-right: 20px;
padding: 0 8px;
background: white;
color: #333;
border: 0 solid white;
Expand All @@ -47,11 +28,38 @@
}
}

.spinner {
.preIcon {
flex-shrink: 0;
width: 20px;
height: 20px;
margin-left: 8px;
}

.spinner {
animation: rotate 2s linear infinite;
}

.clear {
display: flex;
align-items: center;
flex-shrink: 0;
}

.byNameOrId:not(
[_='This `:not` selector is used to increase the specificity of the selector and serves no other purpose.']
) {
flex-shrink: 0;
display: flex;
justify-content: center;
align-items: center;
height: 24px;
margin-left: 16px;
padding: 0 8px;
color: #666;
background: #f0f0f0;
border-radius: 2px;
}

@keyframes rotate {
100% {
transform: rotate(360deg);
Expand Down
Loading

0 comments on commit 7b278f0

Please sign in to comment.