2023-02-10 17:10:13 +03:00
|
|
|
|
import './search.css';
|
|
|
|
|
|
2023-04-29 15:59:51 +03:00
|
|
|
|
import { forwardRef } from 'preact/compat';
|
2023-06-28 18:36:37 +03:00
|
|
|
|
import {
|
|
|
|
|
useEffect,
|
|
|
|
|
useImperativeHandle,
|
|
|
|
|
useLayoutEffect,
|
|
|
|
|
useRef,
|
|
|
|
|
useState,
|
|
|
|
|
} from 'preact/hooks';
|
|
|
|
|
import { InView } from 'react-intersection-observer';
|
2023-02-12 14:36:55 +03:00
|
|
|
|
import { useParams, useSearchParams } from 'react-router-dom';
|
2023-02-10 17:10:13 +03:00
|
|
|
|
|
2023-02-12 14:29:03 +03:00
|
|
|
|
import AccountBlock from '../components/account-block';
|
2023-02-11 12:01:43 +03:00
|
|
|
|
import Icon from '../components/icon';
|
2023-02-10 17:10:13 +03:00
|
|
|
|
import Link from '../components/link';
|
2023-02-11 11:27:52 +03:00
|
|
|
|
import Loader from '../components/loader';
|
2023-04-26 08:09:44 +03:00
|
|
|
|
import NavMenu from '../components/nav-menu';
|
2023-02-10 17:10:13 +03:00
|
|
|
|
import Status from '../components/status';
|
|
|
|
|
import { api } from '../utils/api';
|
2023-02-11 12:57:26 +03:00
|
|
|
|
import useTitle from '../utils/useTitle';
|
2023-02-10 17:10:13 +03:00
|
|
|
|
|
2023-06-28 18:36:37 +03:00
|
|
|
|
const SHORT_LIMIT = 5;
|
|
|
|
|
const LIMIT = 40;
|
|
|
|
|
|
2023-02-18 15:48:24 +03:00
|
|
|
|
function Search(props) {
|
2023-02-12 14:36:55 +03:00
|
|
|
|
const params = useParams();
|
|
|
|
|
const { masto, instance, authenticated } = api({
|
|
|
|
|
instance: params.instance,
|
|
|
|
|
});
|
2023-02-10 19:05:18 +03:00
|
|
|
|
const [uiState, setUiState] = useState('default');
|
2023-04-29 15:59:51 +03:00
|
|
|
|
const [searchParams] = useSearchParams();
|
|
|
|
|
const searchFormRef = useRef();
|
2023-02-18 15:48:24 +03:00
|
|
|
|
const q = props?.query || searchParams.get('q');
|
2023-04-29 15:59:51 +03:00
|
|
|
|
const type = props?.type || searchParams.get('type');
|
|
|
|
|
useTitle(
|
|
|
|
|
q
|
|
|
|
|
? `Search: ${q}${
|
|
|
|
|
type
|
|
|
|
|
? ` (${
|
|
|
|
|
{
|
|
|
|
|
statuses: 'Posts',
|
|
|
|
|
accounts: 'Accounts',
|
|
|
|
|
hashtags: 'Hashtags',
|
|
|
|
|
}[type]
|
|
|
|
|
})`
|
|
|
|
|
: ''
|
|
|
|
|
}`
|
|
|
|
|
: 'Search',
|
|
|
|
|
`/search`,
|
|
|
|
|
);
|
2023-02-11 12:57:26 +03:00
|
|
|
|
|
2023-06-28 18:36:37 +03:00
|
|
|
|
const [showMore, setShowMore] = useState(false);
|
|
|
|
|
const offsetRef = useRef(0);
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
offsetRef.current = 0;
|
|
|
|
|
}, [type]);
|
|
|
|
|
|
|
|
|
|
const scrollableRef = useRef();
|
|
|
|
|
useLayoutEffect(() => {
|
|
|
|
|
scrollableRef.current?.scrollTo?.(0, 0);
|
|
|
|
|
}, [q, type]);
|
|
|
|
|
|
2023-02-10 17:10:13 +03:00
|
|
|
|
const [statusResults, setStatusResults] = useState([]);
|
|
|
|
|
const [accountResults, setAccountResults] = useState([]);
|
2023-02-10 19:05:18 +03:00
|
|
|
|
const [hashtagResults, setHashtagResults] = useState([]);
|
2023-06-28 18:36:37 +03:00
|
|
|
|
|
|
|
|
|
function loadResults(firstLoad) {
|
|
|
|
|
setUiState('loading');
|
|
|
|
|
if (firstLoad && !type) {
|
|
|
|
|
setStatusResults(statusResults.slice(0, SHORT_LIMIT));
|
|
|
|
|
setAccountResults(accountResults.slice(0, SHORT_LIMIT));
|
|
|
|
|
setHashtagResults(hashtagResults.slice(0, SHORT_LIMIT));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
|
const params = {
|
|
|
|
|
q,
|
|
|
|
|
resolve: authenticated,
|
|
|
|
|
limit: SHORT_LIMIT,
|
|
|
|
|
};
|
|
|
|
|
if (type) {
|
|
|
|
|
params.limit = LIMIT;
|
|
|
|
|
params.type = type;
|
|
|
|
|
params.offset = offsetRef.current;
|
|
|
|
|
}
|
|
|
|
|
try {
|
|
|
|
|
const results = await masto.v2.search(params);
|
|
|
|
|
console.log(results);
|
2023-06-30 04:48:52 +03:00
|
|
|
|
if (type && !firstLoad) {
|
2023-06-28 18:36:37 +03:00
|
|
|
|
if (type === 'statuses') {
|
|
|
|
|
setStatusResults((prev) => [...prev, ...results.statuses]);
|
|
|
|
|
} else if (type === 'accounts') {
|
|
|
|
|
setAccountResults((prev) => [...prev, ...results.accounts]);
|
|
|
|
|
} else if (type === 'hashtags') {
|
|
|
|
|
setHashtagResults((prev) => [...prev, ...results.hashtags]);
|
|
|
|
|
}
|
|
|
|
|
offsetRef.current = offsetRef.current + LIMIT;
|
|
|
|
|
setShowMore(!!results[type]?.length);
|
|
|
|
|
} else {
|
|
|
|
|
setStatusResults(results.statuses);
|
|
|
|
|
setAccountResults(results.accounts);
|
|
|
|
|
setHashtagResults(results.hashtags);
|
2023-07-13 15:10:39 +03:00
|
|
|
|
if (type) {
|
|
|
|
|
offsetRef.current = LIMIT;
|
2023-07-13 18:12:05 +03:00
|
|
|
|
setShowMore(!!results[type]?.length);
|
2023-07-13 15:10:39 +03:00
|
|
|
|
}
|
2023-06-28 18:36:37 +03:00
|
|
|
|
}
|
|
|
|
|
setUiState('default');
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error(err);
|
|
|
|
|
setUiState('error');
|
|
|
|
|
}
|
|
|
|
|
})();
|
|
|
|
|
}
|
|
|
|
|
|
2023-02-10 17:10:13 +03:00
|
|
|
|
useEffect(() => {
|
2023-04-29 15:59:51 +03:00
|
|
|
|
// searchFieldRef.current?.focus?.();
|
|
|
|
|
// searchFormRef.current?.focus?.();
|
2023-02-10 17:10:13 +03:00
|
|
|
|
if (q) {
|
2023-04-29 15:59:51 +03:00
|
|
|
|
// searchFieldRef.current.value = q;
|
|
|
|
|
searchFormRef.current?.setValue?.(q);
|
2023-06-28 18:36:37 +03:00
|
|
|
|
loadResults(true);
|
2023-02-10 17:10:13 +03:00
|
|
|
|
}
|
2023-04-29 15:59:51 +03:00
|
|
|
|
}, [q, type, instance]);
|
2023-02-10 17:10:13 +03:00
|
|
|
|
|
|
|
|
|
return (
|
2023-06-28 18:36:37 +03:00
|
|
|
|
<div id="search-page" class="deck-container" ref={scrollableRef}>
|
2023-02-10 17:10:13 +03:00
|
|
|
|
<div class="timeline-deck deck">
|
|
|
|
|
<header>
|
|
|
|
|
<div class="header-grid">
|
|
|
|
|
<div class="header-side">
|
2023-04-26 08:09:44 +03:00
|
|
|
|
<NavMenu />
|
2023-02-10 17:10:13 +03:00
|
|
|
|
</div>
|
2023-04-29 15:59:51 +03:00
|
|
|
|
<SearchForm ref={searchFormRef} />
|
|
|
|
|
<div class="header-side"> </div>
|
2023-02-10 17:10:13 +03:00
|
|
|
|
</div>
|
|
|
|
|
</header>
|
|
|
|
|
<main>
|
2023-04-29 15:59:51 +03:00
|
|
|
|
{!!q && (
|
|
|
|
|
<div class="filter-bar">
|
|
|
|
|
{!!type && <Link to={`/search${q ? `?q=${q}` : ''}`}>‹ All</Link>}
|
|
|
|
|
{[
|
|
|
|
|
{
|
|
|
|
|
label: 'Accounts',
|
|
|
|
|
type: 'accounts',
|
|
|
|
|
to: `/search?q=${q}&type=accounts`,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'Hashtags',
|
|
|
|
|
type: 'hashtags',
|
|
|
|
|
to: `/search?q=${q}&type=hashtags`,
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: 'Posts',
|
|
|
|
|
type: 'statuses',
|
|
|
|
|
to: `/search?q=${q}&type=statuses`,
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
.sort((a, b) => {
|
|
|
|
|
if (a.type === type) return -1;
|
|
|
|
|
if (b.type === type) return 1;
|
|
|
|
|
return 0;
|
|
|
|
|
})
|
|
|
|
|
.map((link) => (
|
|
|
|
|
<Link to={link.to}>{link.label}</Link>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2023-06-28 18:36:37 +03:00
|
|
|
|
{!!q ? (
|
2023-02-10 19:05:18 +03:00
|
|
|
|
<>
|
2023-04-29 15:59:51 +03:00
|
|
|
|
{(!type || type === 'accounts') && (
|
|
|
|
|
<>
|
|
|
|
|
{type !== 'accounts' && (
|
|
|
|
|
<h2 class="timeline-header">Accounts</h2>
|
|
|
|
|
)}
|
|
|
|
|
{accountResults.length > 0 ? (
|
|
|
|
|
<>
|
|
|
|
|
<ul class="timeline flat accounts-list">
|
|
|
|
|
{accountResults.map((account) => (
|
2023-06-30 04:48:52 +03:00
|
|
|
|
<li key={account.id}>
|
2023-04-29 15:59:51 +03:00
|
|
|
|
<AccountBlock
|
|
|
|
|
account={account}
|
|
|
|
|
instance={instance}
|
2023-07-02 13:02:30 +03:00
|
|
|
|
showStats
|
2023-04-29 15:59:51 +03:00
|
|
|
|
/>
|
|
|
|
|
</li>
|
|
|
|
|
))}
|
|
|
|
|
</ul>
|
|
|
|
|
{type !== 'accounts' && (
|
|
|
|
|
<div class="ui-state">
|
|
|
|
|
<Link
|
|
|
|
|
class="plain button"
|
|
|
|
|
to={`/search?q=${q}&type=accounts`}
|
|
|
|
|
>
|
|
|
|
|
See more accounts <Icon icon="arrow-right" />
|
|
|
|
|
</Link>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
</>
|
|
|
|
|
) : (
|
2023-06-30 04:48:52 +03:00
|
|
|
|
!type &&
|
|
|
|
|
(uiState === 'loading' ? (
|
|
|
|
|
<p class="ui-state">
|
|
|
|
|
<Loader abrupt />
|
|
|
|
|
</p>
|
|
|
|
|
) : (
|
|
|
|
|
<p class="ui-state">No accounts found.</p>
|
|
|
|
|
))
|
2023-04-29 15:59:51 +03:00
|
|
|
|
)}
|
|
|
|
|
</>
|
2023-02-10 19:05:18 +03:00
|
|
|
|
)}
|
2023-04-29 15:59:51 +03:00
|
|
|
|
{(!type || type === 'hashtags') && (
|
|
|
|
|
<>
|
|
|
|
|
{type !== 'hashtags' && (
|
|
|
|
|
<h2 class="timeline-header">Hashtags</h2>
|
|
|
|
|
)}
|
|
|
|
|
{hashtagResults.length > 0 ? (
|
|
|
|
|
<>
|
|
|
|
|
<ul class="link-list hashtag-list">
|
|
|
|
|
{hashtagResults.map((hashtag) => (
|
2023-06-30 04:48:52 +03:00
|
|
|
|
<li key={hashtag.name}>
|
2023-04-29 15:59:51 +03:00
|
|
|
|
<Link
|
|
|
|
|
to={
|
|
|
|
|
instance
|
|
|
|
|
? `/${instance}/t/${hashtag.name}`
|
|
|
|
|
: `/t/${hashtag.name}`
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
<Icon icon="hashtag" />
|
|
|
|
|
<span>{hashtag.name}</span>
|
|
|
|
|
</Link>
|
|
|
|
|
</li>
|
|
|
|
|
))}
|
|
|
|
|
</ul>
|
|
|
|
|
{type !== 'hashtags' && (
|
|
|
|
|
<div class="ui-state">
|
|
|
|
|
<Link
|
|
|
|
|
class="plain button"
|
|
|
|
|
to={`/search?q=${q}&type=hashtags`}
|
|
|
|
|
>
|
|
|
|
|
See more hashtags <Icon icon="arrow-right" />
|
|
|
|
|
</Link>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
</>
|
|
|
|
|
) : (
|
2023-06-30 04:48:52 +03:00
|
|
|
|
!type &&
|
|
|
|
|
(uiState === 'loading' ? (
|
|
|
|
|
<p class="ui-state">
|
|
|
|
|
<Loader abrupt />
|
|
|
|
|
</p>
|
|
|
|
|
) : (
|
|
|
|
|
<p class="ui-state">No hashtags found.</p>
|
|
|
|
|
))
|
2023-04-29 15:59:51 +03:00
|
|
|
|
)}
|
|
|
|
|
</>
|
2023-02-10 19:05:18 +03:00
|
|
|
|
)}
|
2023-04-29 15:59:51 +03:00
|
|
|
|
{(!type || type === 'statuses') && (
|
|
|
|
|
<>
|
|
|
|
|
{type !== 'statuses' && (
|
|
|
|
|
<h2 class="timeline-header">Posts</h2>
|
|
|
|
|
)}
|
|
|
|
|
{statusResults.length > 0 ? (
|
|
|
|
|
<>
|
|
|
|
|
<ul class="timeline">
|
|
|
|
|
{statusResults.map((status) => (
|
2023-06-30 04:48:52 +03:00
|
|
|
|
<li key={status.id}>
|
2023-04-29 15:59:51 +03:00
|
|
|
|
<Link
|
|
|
|
|
class="status-link"
|
|
|
|
|
to={
|
|
|
|
|
instance
|
|
|
|
|
? `/${instance}/s/${status.id}`
|
|
|
|
|
: `/s/${status.id}`
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
<Status status={status} />
|
|
|
|
|
</Link>
|
|
|
|
|
</li>
|
|
|
|
|
))}
|
|
|
|
|
</ul>
|
|
|
|
|
{type !== 'statuses' && (
|
|
|
|
|
<div class="ui-state">
|
|
|
|
|
<Link
|
|
|
|
|
class="plain button"
|
|
|
|
|
to={`/search?q=${q}&type=statuses`}
|
|
|
|
|
>
|
|
|
|
|
See more posts <Icon icon="arrow-right" />
|
|
|
|
|
</Link>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
</>
|
|
|
|
|
) : (
|
2023-06-30 04:48:52 +03:00
|
|
|
|
!type &&
|
|
|
|
|
(uiState === 'loading' ? (
|
|
|
|
|
<p class="ui-state">
|
|
|
|
|
<Loader abrupt />
|
|
|
|
|
</p>
|
|
|
|
|
) : (
|
|
|
|
|
<p class="ui-state">No posts found.</p>
|
|
|
|
|
))
|
2023-04-29 15:59:51 +03:00
|
|
|
|
)}
|
|
|
|
|
</>
|
2023-02-10 19:05:18 +03:00
|
|
|
|
)}
|
2023-06-28 18:36:37 +03:00
|
|
|
|
{!!type &&
|
|
|
|
|
(uiState === 'default' ? (
|
|
|
|
|
showMore ? (
|
|
|
|
|
<InView
|
|
|
|
|
onChange={(inView) => {
|
|
|
|
|
if (inView) {
|
|
|
|
|
loadResults();
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
class="plain block"
|
|
|
|
|
onClick={() => loadResults()}
|
|
|
|
|
style={{ marginBlockEnd: '6em' }}
|
|
|
|
|
>
|
|
|
|
|
Show more…
|
|
|
|
|
</button>
|
|
|
|
|
</InView>
|
|
|
|
|
) : (
|
|
|
|
|
<p class="ui-state insignificant">The end.</p>
|
|
|
|
|
)
|
|
|
|
|
) : (
|
|
|
|
|
!!(
|
|
|
|
|
hashtagResults.length ||
|
|
|
|
|
accountResults.length ||
|
|
|
|
|
statusResults.length
|
|
|
|
|
) && (
|
|
|
|
|
<p class="ui-state">
|
|
|
|
|
<Loader abrupt />
|
|
|
|
|
</p>
|
|
|
|
|
)
|
|
|
|
|
))}
|
2023-02-10 19:05:18 +03:00
|
|
|
|
</>
|
2023-02-11 11:27:52 +03:00
|
|
|
|
) : uiState === 'loading' ? (
|
|
|
|
|
<p class="ui-state">
|
|
|
|
|
<Loader abrupt />
|
|
|
|
|
</p>
|
2023-02-10 19:05:18 +03:00
|
|
|
|
) : (
|
2023-02-11 12:57:44 +03:00
|
|
|
|
<p class="ui-state">
|
|
|
|
|
Enter your search term or paste a URL above to get started.
|
|
|
|
|
</p>
|
2023-02-10 17:10:13 +03:00
|
|
|
|
)}
|
|
|
|
|
</main>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default Search;
|
2023-04-29 15:59:51 +03:00
|
|
|
|
|
|
|
|
|
const SearchForm = forwardRef((props, ref) => {
|
|
|
|
|
const { instance } = api();
|
|
|
|
|
const [searchParams, setSearchParams] = useSearchParams();
|
|
|
|
|
const [searchMenuOpen, setSearchMenuOpen] = useState(false);
|
|
|
|
|
const [query, setQuery] = useState(searchParams.q || '');
|
|
|
|
|
const formRef = useRef(null);
|
|
|
|
|
|
|
|
|
|
const searchFieldRef = useRef(null);
|
|
|
|
|
useImperativeHandle(ref, () => ({
|
|
|
|
|
setValue: (value) => {
|
|
|
|
|
setQuery(value);
|
|
|
|
|
},
|
|
|
|
|
focus: () => {
|
|
|
|
|
searchFieldRef.current.focus();
|
|
|
|
|
},
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<form
|
|
|
|
|
ref={formRef}
|
|
|
|
|
class="search-popover-container"
|
|
|
|
|
onSubmit={(e) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
|
|
|
|
if (query) {
|
|
|
|
|
setSearchParams({
|
|
|
|
|
q: query,
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
setSearchParams({});
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<input
|
|
|
|
|
ref={searchFieldRef}
|
|
|
|
|
value={query}
|
|
|
|
|
name="q"
|
|
|
|
|
type="search"
|
|
|
|
|
// autofocus
|
|
|
|
|
placeholder="Search"
|
|
|
|
|
onSearch={(e) => {
|
|
|
|
|
if (!e.target.value) {
|
|
|
|
|
setSearchParams({});
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
onInput={(e) => {
|
|
|
|
|
setQuery(e.target.value);
|
|
|
|
|
setSearchMenuOpen(true);
|
|
|
|
|
}}
|
|
|
|
|
onFocus={() => {
|
|
|
|
|
setSearchMenuOpen(true);
|
|
|
|
|
}}
|
|
|
|
|
onBlur={() => {
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
setSearchMenuOpen(false);
|
|
|
|
|
}, 100);
|
|
|
|
|
formRef.current
|
|
|
|
|
?.querySelector('.search-popover-item.focus')
|
|
|
|
|
?.classList.remove('focus');
|
|
|
|
|
}}
|
|
|
|
|
onKeyDown={(e) => {
|
|
|
|
|
const { key } = e;
|
|
|
|
|
switch (key) {
|
|
|
|
|
case 'Escape':
|
|
|
|
|
setSearchMenuOpen(false);
|
|
|
|
|
break;
|
|
|
|
|
case 'Down':
|
|
|
|
|
case 'ArrowDown':
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
if (searchMenuOpen) {
|
|
|
|
|
const focusItem = formRef.current.querySelector(
|
|
|
|
|
'.search-popover-item.focus',
|
|
|
|
|
);
|
|
|
|
|
if (focusItem) {
|
|
|
|
|
let nextItem = focusItem.nextElementSibling;
|
|
|
|
|
while (nextItem && nextItem.hidden) {
|
|
|
|
|
nextItem = nextItem.nextElementSibling;
|
|
|
|
|
}
|
|
|
|
|
if (nextItem) {
|
|
|
|
|
nextItem.classList.add('focus');
|
|
|
|
|
const siblings = Array.from(
|
|
|
|
|
nextItem.parentElement.children,
|
|
|
|
|
).filter((el) => el !== nextItem);
|
|
|
|
|
siblings.forEach((el) => {
|
|
|
|
|
el.classList.remove('focus');
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
const firstItem = formRef.current.querySelector(
|
|
|
|
|
'.search-popover-item',
|
|
|
|
|
);
|
|
|
|
|
if (firstItem) {
|
|
|
|
|
firstItem.classList.add('focus');
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'Up':
|
|
|
|
|
case 'ArrowUp':
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
if (searchMenuOpen) {
|
|
|
|
|
const focusItem = document.querySelector(
|
|
|
|
|
'.search-popover-item.focus',
|
|
|
|
|
);
|
|
|
|
|
if (focusItem) {
|
|
|
|
|
let prevItem = focusItem.previousElementSibling;
|
|
|
|
|
while (prevItem && prevItem.hidden) {
|
|
|
|
|
prevItem = prevItem.previousElementSibling;
|
|
|
|
|
}
|
|
|
|
|
if (prevItem) {
|
|
|
|
|
prevItem.classList.add('focus');
|
|
|
|
|
const siblings = Array.from(
|
|
|
|
|
prevItem.parentElement.children,
|
|
|
|
|
).filter((el) => el !== prevItem);
|
|
|
|
|
siblings.forEach((el) => {
|
|
|
|
|
el.classList.remove('focus');
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
const lastItem = document.querySelector(
|
|
|
|
|
'.search-popover-item:last-child',
|
|
|
|
|
);
|
|
|
|
|
if (lastItem) {
|
|
|
|
|
lastItem.classList.add('focus');
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case 'Enter':
|
|
|
|
|
if (searchMenuOpen) {
|
|
|
|
|
const focusItem = document.querySelector(
|
|
|
|
|
'.search-popover-item.focus',
|
|
|
|
|
);
|
|
|
|
|
if (focusItem) {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
focusItem.click();
|
|
|
|
|
}
|
|
|
|
|
setSearchMenuOpen(false);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
<div class="search-popover" hidden={!searchMenuOpen || !query}>
|
|
|
|
|
{!!query &&
|
|
|
|
|
[
|
|
|
|
|
{
|
|
|
|
|
label: (
|
|
|
|
|
<>
|
|
|
|
|
Posts with <q>{query}</q>
|
|
|
|
|
</>
|
|
|
|
|
),
|
|
|
|
|
to: `/search?q=${encodeURIComponent(query)}&type=statuses`,
|
|
|
|
|
hidden: /^https?:/.test(query),
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: (
|
|
|
|
|
<>
|
|
|
|
|
Posts tagged with <mark>#{query.replace(/^#/, '')}</mark>
|
|
|
|
|
</>
|
|
|
|
|
),
|
|
|
|
|
to: `/${instance}/t/${query.replace(/^#/, '')}`,
|
|
|
|
|
hidden:
|
|
|
|
|
/^@/.test(query) || /^https?:/.test(query) || /\s/.test(query),
|
|
|
|
|
top: /^#/.test(query),
|
|
|
|
|
type: 'link',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: (
|
|
|
|
|
<>
|
|
|
|
|
Look up <mark>{query}</mark>
|
|
|
|
|
</>
|
|
|
|
|
),
|
|
|
|
|
to: `/${query}`,
|
|
|
|
|
hidden: !/^https?:/.test(query),
|
|
|
|
|
top: /^https?:/.test(query),
|
|
|
|
|
type: 'link',
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
label: (
|
|
|
|
|
<>
|
|
|
|
|
Accounts with <q>{query}</q>
|
|
|
|
|
</>
|
|
|
|
|
),
|
|
|
|
|
to: `/search?q=${encodeURIComponent(query)}&type=accounts`,
|
|
|
|
|
},
|
|
|
|
|
]
|
|
|
|
|
.sort((a, b) => {
|
|
|
|
|
if (a.top && !b.top) return -1;
|
|
|
|
|
if (!a.top && b.top) return 1;
|
|
|
|
|
return 0;
|
|
|
|
|
})
|
|
|
|
|
.map(({ label, to, hidden, type }) => (
|
|
|
|
|
<Link to={to} class="search-popover-item" hidden={hidden}>
|
|
|
|
|
<Icon
|
|
|
|
|
icon={type === 'link' ? 'arrow-right' : 'search'}
|
|
|
|
|
class="more-insignificant"
|
|
|
|
|
/>
|
|
|
|
|
<span>{label}</span>{' '}
|
|
|
|
|
</Link>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</form>
|
|
|
|
|
);
|
|
|
|
|
});
|