Skip to content

Commit

Permalink
[frogend] Emoji categories (#1051)
Browse files Browse the repository at this point in the history
* emoji category combobox

* emoji categorizing

* dropdown entry separation

* emoji filtering/sorting

* add some explaining comments

* remove unneeded default-value code

* remove wrongly created package.json

* configurable ComboBox label+placeHolder
  • Loading branch information
f0x52 committed Nov 16, 2022
1 parent 940abc2 commit aa5c4e0
Show file tree
Hide file tree
Showing 10 changed files with 249 additions and 35 deletions.
2 changes: 1 addition & 1 deletion web/source/css/base.css
Original file line number Diff line number Diff line change
Expand Up @@ -290,7 +290,7 @@ section.error {
font-weight: bold;
}

input, select, textarea {
input, select, textarea, .input {
box-sizing: border-box;
border: 0.15rem solid $input-border;
border-radius: 0.1rem;
Expand Down
4 changes: 4 additions & 0 deletions web/source/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,11 +12,13 @@
},
"dependencies": {
"@reduxjs/toolkit": "^1.8.6",
"ariakit": "^2.0.0-next.41",
"bluebird": "^3.7.2",
"dotty": "^0.1.2",
"is-valid-domain": "^0.1.6",
"js-file-download": "^0.4.12",
"langs": "^2.0.0",
"match-sorter": "^6.3.1",
"modern-normalize": "^1.1.0",
"photoswipe": "^5.3.3",
"photoswipe-dynamic-caption-plugin": "^1.2.7",
Expand All @@ -28,6 +30,8 @@
"redux-devtools-extension": "^2.13.9",
"redux-persist": "^6.0.0",
"skulk": "^0.0.6",
"split-filter-n": "^1.1.3",
"syncpipe": "^1.0.0",
"wouter": "^2.8.0-alpha.2"
},
"devDependencies": {
Expand Down
49 changes: 41 additions & 8 deletions web/source/settings/admin/emoji/new-emoji.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,44 +20,68 @@

const Promise = require('bluebird');
const React = require("react");
const { matchSorter } = require("match-sorter");

const FakeToot = require("../../components/fake-toot");
const MutateButton = require("../../components/mutation-button");
const ComboBox = require("../../components/combo-box");

const {
const {
useTextInput,
useFileInput
useFileInput,
useComboBoxInput
} = require("../../components/form");

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

module.exports = function NewEmojiForm({emoji}) {
module.exports = function NewEmojiForm({ emoji, emojiByCategory }) {
const emojiCodes = React.useMemo(() => {
return new Set(emoji.map((e) => e.shortcode));
}, [emoji]);

const [addEmoji, result] = query.useAddEmojiMutation();

const [onFileChange, resetFile, {image, imageURL, imageInfo}] = useFileInput("image", {
const [onFileChange, resetFile, { image, imageURL, imageInfo }] = useFileInput("image", {
withPreview: true,
maxSize: 50 * 1024
});

const [onShortcodeChange, resetShortcode, {shortcode, setShortcode, shortcodeRef}] = useTextInput("shortcode", {
const [onShortcodeChange, resetShortcode, { shortcode, setShortcode, shortcodeRef }] = useTextInput("shortcode", {
validator: function validateShortcode(code) {
return emojiCodes.has(code)
? "Shortcode already in use"
: "";
}
});

const [categoryState, resetCategory, { category }] = useComboBoxInput("category");

// 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(_, category), // 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, category]);

React.useEffect(() => {
if (shortcode.length == 0) {
if (image != undefined) {
let [name, _ext] = image.name.split(".");
setShortcode(name);
}
}
// we explicitly don't want to add 'shortcode' as a dependency here
// because we only want this to update to the filename if the field is empty
// at the moment the file is selected, not some time after when the field is emptied
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [image]);

Expand All @@ -69,11 +93,13 @@ module.exports = function NewEmojiForm({emoji}) {
Promise.try(() => {
return addEmoji({
image,
shortcode
shortcode,
category
});
}).then(() => {
resetFile();
resetShortcode();
resetCategory();
});
}

Expand Down Expand Up @@ -125,8 +151,15 @@ module.exports = function NewEmojiForm({emoji}) {
value={shortcode}
/>
</div>

<MutateButton text="Upload emoji" result={result}/>

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

<MutateButton text="Upload emoji" result={result} />
</form>
</div>
);
Expand Down
33 changes: 14 additions & 19 deletions web/source/settings/admin/emoji/overview.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@

const React = require("react");
const {Link} = require("wouter");
const defaultValue = require('default-value');
const splitFilterN = require("split-filter-n");

const NewEmojiForm = require("./new-emoji");

Expand All @@ -30,11 +30,18 @@ const base = "/settings/admin/custom-emoji";

module.exports = function EmojiOverview() {
const {
data: emoji,
data: emoji = [],
isLoading,
error
} = query.useGetAllEmojiQuery({filter: "domain:local"});

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

return (
<>
<h1>Custom Emoji</h1>
Expand All @@ -44,33 +51,21 @@ module.exports = function EmojiOverview() {
{isLoading
? "Loading..."
: <>
<EmojiList emoji={emoji}/>
<NewEmojiForm emoji={emoji}/>
<EmojiList emoji={emoji} emojiByCategory={emojiByCategory}/>
<NewEmojiForm emoji={emoji} emojiByCategory={emojiByCategory}/>
</>
}
</>
);
};

function EmojiList({emoji}) {
const byCategory = React.useMemo(() => {
const categories = {};

emoji.forEach((emoji) => {
let cat = defaultValue(emoji.category, "Unsorted");
categories[cat] = defaultValue(categories[cat], []);
categories[cat].push(emoji);
});

return categories;
}, [emoji]);

function EmojiList({emoji, emojiByCategory}) {
return (
<div>
<h2>Overview</h2>
<div className="list emoji-list">
{emoji.length == 0 && "No local emoji yet"}
{Object.entries(byCategory).map(([category, entries]) => {
{emoji.length == 0 && "No local emoji yet, add one below"}
{Object.entries(emojiByCategory).map(([category, entries]) => {
return <EmojiCategory key={category} category={category} entries={entries}/>;
})}
</div>
Expand Down
49 changes: 49 additions & 0 deletions web/source/settings/components/combo-box.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
/*
GoToSocial
Copyright (C) 2021-2022 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 {
Combobox,
ComboboxItem,
ComboboxPopover,
} = require("ariakit/combobox");

module.exports = function ComboBox({state, items, label, placeHolder}) {
return (
<div className="form-field combobox-wrapper">
<label>
{label}
<Combobox
state={state}
placeholder={placeHolder}
className="combobox input"
/>
</label>
<ComboboxPopover state={state} className="popover">
{items.map(([key, value]) => (
<ComboboxItem className="combobox-item" key={key} value={key}>
{value}
</ComboboxItem>
))}
</ComboboxPopover>
</div>
);
};
37 changes: 37 additions & 0 deletions web/source/settings/components/form/combobox.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
/*
GoToSocial
Copyright (C) 2021-2022 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 { useComboboxState } = require("ariakit/combobox");

module.exports = function useComboBoxInput({name, Name}, {validator} = {}) {
const state = useComboboxState({ gutter: 0, sameWidth: true });

function reset() {
state.value = "";
}

return [
state,
reset,
{
[name]: state.value,
}
];
};
3 changes: 2 additions & 1 deletion web/source/settings/components/form/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -32,5 +32,6 @@ function makeHook(func) {

module.exports = {
useTextInput: makeHook(require("./text")),
useFileInput: makeHook(require("./file"))
useFileInput: makeHook(require("./file")),
useComboBoxInput: makeHook(require("./combobox"))
};
9 changes: 4 additions & 5 deletions web/source/settings/redux/reducers/user.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@

const { createSlice } = require("@reduxjs/toolkit");
const d = require("dotty");
const defaultValue = require("default-value");

module.exports = createSlice({
name: "user",
Expand All @@ -30,10 +29,10 @@ module.exports = createSlice({
},
reducers: {
setAccount: (state, { payload }) => {
payload.source = defaultValue(payload.source, {});
payload.source.language = defaultValue(payload.source.language.toUpperCase(), "EN");
payload.source.status_format = defaultValue(payload.source.status_format, "plain");
payload.source.sensitive = defaultValue(payload.source.sensitive, false);
payload.source = payload.source ?? {};
payload.source.language = payload.source.language.toUpperCase() ?? "EN";
payload.source.status_format = payload.source.status_format ?? "plain";
payload.source.sensitive = payload.source.sensitive ?? false;

state.profile = payload;
// /user/settings only needs a copy of the 'source' obj
Expand Down
58 changes: 58 additions & 0 deletions web/source/settings/style.css
Original file line number Diff line number Diff line change
Expand Up @@ -502,4 +502,62 @@ span.form-info {
.instance-list .filter {
flex-direction: column;
}
}

.combobox-wrapper {
display: flex;
flex-direction: column;

input[aria-expanded="true"] {
border-bottom: none;
}
}

.combobox {
height: 2.5rem;
font-size: 1rem;
line-height: 1.5rem;
}

.popover {
position: relative;
z-index: 50;
display: flex;
max-height: min(var(--popover-available-height,300px),300px);
flex-direction: column;
overflow: auto;
overscroll-behavior: contain;
border: 0.15rem solid $orange2;
background: $bg-accent;
}

.combobox-item {
display: flex;
cursor: pointer;
scroll-margin: 0.5rem;
align-items: center;
gap: 0.5rem;
padding: 0.5rem;
line-height: 1.5rem;
border-bottom: 0.15rem solid $gray3;

&:last-child {
border: none;
}

img {
height: 1.5rem;
width: 1.5rem;
object-fit: contain;
}
}

.combobox-item:hover {
background: $button-hover-bg;
color: $button-fg;
}

.combobox-item[data-active-item] {
background: $button-hover-bg;
color: hsl(204 20% 100%);
}

0 comments on commit aa5c4e0

Please sign in to comment.