summaryrefslogtreecommitdiff
path: root/web/source/settings/admin/emoji/category-select.jsx
blob: d22534ea8d26f5d6aae291b2e8f3e37e1159f330 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
/*
	 GoToSocial
	 Copyright (C) 2021-2023 GoToSocial Authors admin@gotosocial.org

	 This program is free software: you can redistribute it and/or modify
	 it under the terms of the GNU Affero General Public License as published by
	 the Free Software Foundation, either version 3 of the License, or
	 (at your option) any later version.

	 This program is distributed in the hope that it will be useful,
	 but WITHOUT ANY WARRANTY; without even the implied warranty of
	 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
	 GNU Affero General Public License for more details.

	 You should have received a copy of the GNU Affero General Public License
	 along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

"use strict";

const React = require("react");
const splitFilterN = require("split-filter-n");
const syncpipe = require('syncpipe');
const { matchSorter } = require("match-sorter");

const query = require("../../lib/query");

const ComboBox = require("../../components/combo-box");

function useEmojiByCategory(emoji) {
	// split all emoji over an object keyed by the category names (or Unsorted)
	return React.useMemo(() => splitFilterN(
		emoji,
		[],
		(entry) => entry.category ?? "Unsorted"
	), [emoji]);
}

function CategorySelect({value, categoryState, setIsNew=() => {}, children}) {
	const {
		data: emoji = [],
		isLoading,
		isSuccess,
		error
	} = query.useGetAllEmojiQuery({filter: "domain:local"});

	const emojiByCategory = useEmojiByCategory(emoji);

	const categories = React.useMemo(() => new Set(Object.keys(emojiByCategory)), [emojiByCategory]);

	// data used by the ComboBox element to select an emoji category
	const categoryItems = React.useMemo(() => {
		return syncpipe(emojiByCategory, [
			(_) => Object.keys(_),            // just emoji category names
			(_) => matchSorter(_, value, {threshold: matchSorter.rankings.NO_MATCH}),  // sorted by complex algorithm
			(_) => _.map((categoryName) => [  // map to input value, and selectable element with icon
				categoryName,
				<>
					<img src={emojiByCategory[categoryName][0].static_url} aria-hidden="true"></img>
					{categoryName}
				</>
			])
		]);
	}, [emojiByCategory, value]);

	React.useEffect(() => {
		if (value != undefined && isSuccess && value.trim().length > 0) {
			setIsNew(!categories.has(value.trim()));
		}
	}, [categories, value, setIsNew, isSuccess]);

	if (error) { // fall back to plain text input, but this would almost certainly have caused a bigger error message elsewhere
		return (
			<>
				<input type="text" placeholder="e.g., reactions" onChange={(e) => {categoryState.value = e.target.value;}}/>;
			</>
		);
	} else if (isLoading) {
		return <input type="text" value="Loading categories..." disabled={true}/>;
	}

	return (
		<ComboBox
			state={categoryState}
			items={categoryItems}
			label="Category"
			placeHolder="e.g., reactions"
			children={children}
		/>
	);
}

module.exports = {
	useEmojiByCategory,
	CategorySelect
};