frontend-rw #1

Merged
marcin merged 347 commits from frontend-rw into develop 2024-12-05 15:32:18 -08:00
10 changed files with 43 additions and 113 deletions
Showing only changes of commit 6c61f749fa - Show all commits

View file

@ -679,7 +679,7 @@ const selectComposeSuggestion = (composeId: string, position: number, token: str
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
let completion = '', startPosition = position; let completion = '', startPosition = position;
if (typeof suggestion === 'object' && suggestion.id) { if (typeof suggestion === 'object' && 'id' in suggestion) {
completion = isNativeEmoji(suggestion) ? suggestion.native : suggestion.colons; completion = isNativeEmoji(suggestion) ? suggestion.native : suggestion.colons;
startPosition = position - 1; startPosition = position - 1;

View file

@ -9,10 +9,6 @@ import { STATUS_FETCH_SOURCE_FAIL, STATUS_FETCH_SOURCE_REQUEST, STATUS_FETCH_SOU
import type { Account, CreateEventParams, Location, MediaAttachment, PaginatedResponse, Status } from 'pl-api'; import type { Account, CreateEventParams, Location, MediaAttachment, PaginatedResponse, Status } from 'pl-api';
import type { AppDispatch, RootState } from 'pl-fe/store'; import type { AppDispatch, RootState } from 'pl-fe/store';
const LOCATION_SEARCH_REQUEST = 'LOCATION_SEARCH_REQUEST' as const;
const LOCATION_SEARCH_SUCCESS = 'LOCATION_SEARCH_SUCCESS' as const;
const LOCATION_SEARCH_FAIL = 'LOCATION_SEARCH_FAIL' as const;
const EVENT_SUBMIT_REQUEST = 'EVENT_SUBMIT_REQUEST' as const; const EVENT_SUBMIT_REQUEST = 'EVENT_SUBMIT_REQUEST' as const;
const EVENT_SUBMIT_SUCCESS = 'EVENT_SUBMIT_SUCCESS' as const; const EVENT_SUBMIT_SUCCESS = 'EVENT_SUBMIT_SUCCESS' as const;
const EVENT_SUBMIT_FAIL = 'EVENT_SUBMIT_FAIL' as const; const EVENT_SUBMIT_FAIL = 'EVENT_SUBMIT_FAIL' as const;
@ -73,18 +69,6 @@ const messages = defineMessages({
rejected: { id: 'compose_event.participation_requests.reject_success', defaultMessage: 'User rejected' }, rejected: { id: 'compose_event.participation_requests.reject_success', defaultMessage: 'User rejected' },
}); });
const locationSearch = (query: string, signal?: AbortSignal) =>
(dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: LOCATION_SEARCH_REQUEST, query });
return getClient(getState).search.searchLocation(query, { signal }).then((locations) => {
dispatch({ type: LOCATION_SEARCH_SUCCESS, locations });
return locations;
}).catch(error => {
dispatch({ type: LOCATION_SEARCH_FAIL });
throw error;
});
};
const submitEvent = ({ const submitEvent = ({
statusId, statusId,
name, name,
@ -509,9 +493,6 @@ type EventsAction =
| EventFormSetAction; | EventFormSetAction;
export { export {
LOCATION_SEARCH_REQUEST,
LOCATION_SEARCH_SUCCESS,
LOCATION_SEARCH_FAIL,
EVENT_SUBMIT_REQUEST, EVENT_SUBMIT_REQUEST,
EVENT_SUBMIT_SUCCESS, EVENT_SUBMIT_SUCCESS,
EVENT_SUBMIT_FAIL, EVENT_SUBMIT_FAIL,
@ -547,7 +528,6 @@ export {
JOINED_EVENTS_FETCH_REQUEST, JOINED_EVENTS_FETCH_REQUEST,
JOINED_EVENTS_FETCH_SUCCESS, JOINED_EVENTS_FETCH_SUCCESS,
JOINED_EVENTS_FETCH_FAIL, JOINED_EVENTS_FETCH_FAIL,
locationSearch,
submitEvent, submitEvent,
submitEventRequest, submitEventRequest,
submitEventSuccess, submitEventSuccess,

View file

@ -0,0 +1,16 @@
import { useQuery } from '@tanstack/react-query';
import { useClient } from 'pl-fe/hooks/use-client';
const useSearchLocation = (query: string) => {
const client = useClient();
return useQuery({
queryKey: ['search', 'location', query],
queryFn: ({ signal }) => client.search.searchLocation(query, { signal }),
gcTime: 60 * 1000,
enabled: !!query.trim(),
});
};
export { useSearchLocation };

View file

@ -8,15 +8,18 @@ import Portal from 'pl-fe/components/ui/portal';
import AutosuggestAccount from 'pl-fe/features/compose/components/autosuggest-account'; import AutosuggestAccount from 'pl-fe/features/compose/components/autosuggest-account';
import { textAtCursorMatchesToken } from 'pl-fe/utils/suggestions'; import { textAtCursorMatchesToken } from 'pl-fe/utils/suggestions';
import AutosuggestLocation from './autosuggest-location';
import type { Location } from 'pl-api';
import type { Menu, MenuItem } from 'pl-fe/components/dropdown-menu'; import type { Menu, MenuItem } from 'pl-fe/components/dropdown-menu';
import type { InputThemes } from 'pl-fe/components/ui/input'; import type { InputThemes } from 'pl-fe/components/ui/input';
import type { Emoji } from 'pl-fe/features/emoji'; import type { Emoji } from 'pl-fe/features/emoji';
type AutoSuggestion = string | Emoji; type AutoSuggestion = string | Emoji | Location;
interface IAutosuggestInput extends Pick<React.HTMLAttributes<HTMLInputElement>, 'onChange' | 'onKeyUp' | 'onKeyDown'> { interface IAutosuggestInput extends Pick<React.HTMLAttributes<HTMLInputElement>, 'onChange' | 'onKeyUp' | 'onKeyDown'> {
value: string; value: string;
suggestions: Array<any>; suggestions: Array<AutoSuggestion>;
disabled?: boolean; disabled?: boolean;
placeholder?: string; placeholder?: string;
onSuggestionSelected: (tokenStart: number, lastToken: string | null, suggestion: AutoSuggestion) => void; onSuggestionSelected: (tokenStart: number, lastToken: string | null, suggestion: AutoSuggestion) => void;
@ -29,7 +32,6 @@ interface IAutosuggestInput extends Pick<React.HTMLAttributes<HTMLInputElement>,
searchTokens?: string[]; searchTokens?: string[];
maxLength?: number; maxLength?: number;
menu?: Menu; menu?: Menu;
renderSuggestion?: React.FC<{ id: string }>;
hidePortal?: boolean; hidePortal?: boolean;
theme?: InputThemes; theme?: InputThemes;
} }
@ -165,16 +167,12 @@ const AutosuggestInput: React.FC<IAutosuggestInput> = ({
const renderSuggestion = (suggestion: AutoSuggestion, i: number) => { const renderSuggestion = (suggestion: AutoSuggestion, i: number) => {
let inner, key; let inner, key;
if (props.renderSuggestion && typeof suggestion === 'string') { if (typeof suggestion === 'object' && 'origin_id' in suggestion) {
const RenderSuggestion = props.renderSuggestion; inner = <AutosuggestLocation location={suggestion} />;
inner = <RenderSuggestion id={suggestion} />; key = suggestion.origin_id;
key = suggestion;
} else if (typeof suggestion === 'object') { } else if (typeof suggestion === 'object') {
inner = <AutosuggestEmoji emoji={suggestion} />; inner = <AutosuggestEmoji emoji={suggestion} />;
key = suggestion.id; key = suggestion.id;
} else if (suggestion[0] === '#') {
inner = suggestion;
key = suggestion;
} else { } else {
inner = <AutosuggestAccount id={suggestion} />; inner = <AutosuggestAccount id={suggestion} />;
key = suggestion; key = suggestion;

View file

@ -4,7 +4,8 @@ import HStack from 'pl-fe/components/ui/hstack';
import Icon from 'pl-fe/components/ui/icon'; import Icon from 'pl-fe/components/ui/icon';
import Stack from 'pl-fe/components/ui/stack'; import Stack from 'pl-fe/components/ui/stack';
import Text from 'pl-fe/components/ui/text'; import Text from 'pl-fe/components/ui/text';
import { useAppSelector } from 'pl-fe/hooks/use-app-selector';
import type { Location } from 'pl-api';
const buildingCommunityIcon = require('@tabler/icons/outline/building-community.svg'); const buildingCommunityIcon = require('@tabler/icons/outline/building-community.svg');
const homeIcon = require('@tabler/icons/outline/home-2.svg'); const homeIcon = require('@tabler/icons/outline/home-2.svg');
@ -21,12 +22,10 @@ const ADDRESS_ICONS: Record<string, string> = {
}; };
interface IAutosuggestLocation { interface IAutosuggestLocation {
id: string; location: Location;
} }
const AutosuggestLocation: React.FC<IAutosuggestLocation> = ({ id }) => { const AutosuggestLocation: React.FC<IAutosuggestLocation> = ({ location }) => {
const location = useAppSelector((state) => state.locations.get(id));
if (!location) return null; if (!location) return null;
return ( return (

View file

@ -1,14 +1,13 @@
import { useDebounce } from '@uidotdev/usehooks';
import clsx from 'clsx'; import clsx from 'clsx';
import throttle from 'lodash/throttle'; import React, { useState } from 'react';
import React, { useCallback, useEffect, useRef, useState } from 'react';
import { defineMessages, useIntl } from 'react-intl'; import { defineMessages, useIntl } from 'react-intl';
import { locationSearch } from 'pl-fe/actions/events'; import { useSearchLocation } from 'pl-fe/api/hooks/search/use-search-location';
import AutosuggestInput, { AutoSuggestion } from 'pl-fe/components/autosuggest-input'; import AutosuggestInput, { AutoSuggestion } from 'pl-fe/components/autosuggest-input';
import Icon from 'pl-fe/components/icon'; import Icon from 'pl-fe/components/icon';
import { useAppDispatch } from 'pl-fe/hooks/use-app-dispatch';
import AutosuggestLocation from './autosuggest-location'; import type { Location } from 'pl-api';
const noOp = () => {}; const noOp = () => {};
@ -17,27 +16,24 @@ const messages = defineMessages({
}); });
interface ILocationSearch { interface ILocationSearch {
onSelected: (locationId: string) => void; onSelected: (location: Location) => void;
} }
const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => { const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => {
const intl = useIntl(); const intl = useIntl();
const dispatch = useAppDispatch();
const [locationIds, setLocationIds] = useState<Array<string>>([]);
const controller = useRef(new AbortController());
const [value, setValue] = useState(''); const [value, setValue] = useState('');
const debouncedValue = useDebounce(value, 400);
const locationsQuery = useSearchLocation(debouncedValue);
const empty = !(value.length > 0); const empty = !(value.length > 0);
const handleChange: React.ChangeEventHandler<HTMLInputElement> = ({ target }) => { const handleChange: React.ChangeEventHandler<HTMLInputElement> = ({ target }) => {
refreshCancelToken();
handleLocationSearch(target.value);
setValue(target.value); setValue(target.value);
}; };
const handleSelected = (_tokenStart: number, _lastToken: string | null, suggestion: AutoSuggestion) => { const handleSelected = (_tokenStart: number, _lastToken: string | null, suggestion: AutoSuggestion) => {
if (typeof suggestion === 'string') { if (typeof suggestion === 'object' && 'origin_id' in suggestion) {
onSelected(suggestion); onSelected(suggestion);
} }
}; };
@ -56,30 +52,6 @@ const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => {
} }
}; };
const refreshCancelToken = () => {
controller.current.abort();
controller.current = new AbortController();
};
const clearResults = () => {
setLocationIds([]);
};
const handleLocationSearch = useCallback(throttle(q => {
dispatch(locationSearch(q, controller.current.signal))
.then((locations: { origin_id: string }[]) => {
const locationIds = locations.map(location => location.origin_id);
setLocationIds(locationIds);
})
.catch(noOp);
}, 900, { leading: true, trailing: true }), []);
useEffect(() => {
if (value === '') {
clearResults();
}
}, [value]);
return ( return (
<div className='relative'> <div className='relative'>
<AutosuggestInput <AutosuggestInput
@ -87,13 +59,12 @@ const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => {
placeholder={intl.formatMessage(messages.placeholder)} placeholder={intl.formatMessage(messages.placeholder)}
value={value} value={value}
onChange={handleChange} onChange={handleChange}
suggestions={locationIds} suggestions={locationsQuery.data || []}
onSuggestionsFetchRequested={noOp} onSuggestionsFetchRequested={noOp}
onSuggestionsClearRequested={noOp} onSuggestionsClearRequested={noOp}
onSuggestionSelected={handleSelected} onSuggestionSelected={handleSelected}
searchTokens={[]} searchTokens={[]}
onKeyDown={handleKeyDown} onKeyDown={handleKeyDown}
renderSuggestion={AutosuggestLocation}
/> />
<div role='button' tabIndex={0} className='absolute inset-y-0 right-0 flex cursor-pointer items-center px-3 rtl:left-0 rtl:right-auto' onClick={handleClear}> <div role='button' tabIndex={0} className='absolute inset-y-0 right-0 flex cursor-pointer items-center px-3 rtl:left-0 rtl:right-auto' onClick={handleClear}>
<Icon src={require('@tabler/icons/outline/search.svg')} className={clsx('size-5 text-gray-600', { 'hidden': !empty })} /> <Icon src={require('@tabler/icons/outline/search.svg')} className={clsx('size-5 text-gray-600', { 'hidden': !empty })} />

View file

@ -93,16 +93,8 @@ const EditEvent: React.FC<IEditEvent> = ({ statusId }) => {
setApprovalRequired(target.checked); setApprovalRequired(target.checked);
}; };
const onChangeLocation = (value: string | null) => { const onChangeLocation = (location: Location | null) => {
dispatch((_, getState) => {
let location = null;
if (value) {
location = getState().locations.get(value, null);
}
setLocation(location); setLocation(location);
});
}; };
const handleFiles = (files: FileList) => { const handleFiles = (files: FileList) => {

View file

@ -44,7 +44,9 @@ import AutosuggestAccount from '../../components/autosuggest-account';
import { $createEmojiNode } from '../nodes/emoji-node'; import { $createEmojiNode } from '../nodes/emoji-node';
import { $createMentionNode } from '../nodes/mention-node'; import { $createMentionNode } from '../nodes/mention-node';
import type { AutoSuggestion } from 'pl-fe/components/autosuggest-input'; import type { Emoji } from 'pl-fe/features/emoji';
type AutoSuggestion = string | Emoji;
type QueryMatch = { type QueryMatch = {
leadOffset: number; leadOffset: number;

View file

@ -23,7 +23,6 @@ import instance from './instance';
import listAdder from './list-adder'; import listAdder from './list-adder';
import listEditor from './list-editor'; import listEditor from './list-editor';
import lists from './lists'; import lists from './lists';
import locations from './locations';
import me from './me'; import me from './me';
import meta from './meta'; import meta from './meta';
import notifications from './notifications'; import notifications from './notifications';
@ -65,7 +64,6 @@ const reducers = {
listAdder, listAdder,
listEditor, listEditor,
lists, lists,
locations,
me, me,
meta, meta,
notifications, notifications,

View file

@ -1,26 +0,0 @@
import { Map as ImmutableMap } from 'immutable';
import { Location } from 'pl-api';
import { AnyAction } from 'redux';
import { LOCATION_SEARCH_SUCCESS } from 'pl-fe/actions/events';
type State = ImmutableMap<any, Location>;
const initialState: State = ImmutableMap();
const normalizeLocations = (state: State, locations: Array<Location>) =>
locations.reduce(
(state: State, location: Location) => state.set(location.origin_id, location),
state,
);
const locations = (state: State = initialState, action: AnyAction): State => {
switch (action.type) {
case LOCATION_SEARCH_SUCCESS:
return normalizeLocations(state, action.locations);
default:
return state;
}
};
export { locations as default };