2023-02-16 12:51:54 +03:00
|
|
|
import './shortcuts-settings.css';
|
|
|
|
|
|
|
|
import { useEffect, useState } from 'preact/hooks';
|
|
|
|
import { useSnapshot } from 'valtio';
|
|
|
|
|
|
|
|
import { api } from '../utils/api';
|
|
|
|
import states from '../utils/states';
|
|
|
|
|
|
|
|
import AsyncText from './AsyncText';
|
|
|
|
import Icon from './icon';
|
|
|
|
|
2023-02-18 14:39:17 +03:00
|
|
|
const SHORTCUTS_LIMIT = 9;
|
|
|
|
|
2023-02-16 12:51:54 +03:00
|
|
|
const TYPES = [
|
|
|
|
'following',
|
|
|
|
'notifications',
|
|
|
|
'list',
|
|
|
|
'public',
|
2023-02-18 15:48:24 +03:00
|
|
|
// NOTE: Hide for now
|
|
|
|
// 'search', // Search on Mastodon ain't great
|
|
|
|
// 'account-statuses', // Need @acct search first
|
2023-02-16 12:51:54 +03:00
|
|
|
'bookmarks',
|
|
|
|
'favourites',
|
|
|
|
'hashtag',
|
|
|
|
];
|
|
|
|
const TYPE_TEXT = {
|
2023-02-18 15:48:24 +03:00
|
|
|
following: 'Home / Following',
|
2023-02-16 12:51:54 +03:00
|
|
|
notifications: 'Notifications',
|
|
|
|
list: 'List',
|
|
|
|
public: 'Public',
|
|
|
|
search: 'Search',
|
|
|
|
'account-statuses': 'Account',
|
|
|
|
bookmarks: 'Bookmarks',
|
|
|
|
favourites: 'Favourites',
|
|
|
|
hashtag: 'Hashtag',
|
|
|
|
};
|
|
|
|
const TYPE_PARAMS = {
|
|
|
|
list: [
|
|
|
|
{
|
|
|
|
text: 'List ID',
|
|
|
|
name: 'id',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
public: [
|
|
|
|
{
|
|
|
|
text: 'Local only',
|
|
|
|
name: 'local',
|
|
|
|
type: 'checkbox',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
text: 'Instance',
|
|
|
|
name: 'instance',
|
|
|
|
type: 'text',
|
|
|
|
placeholder: 'e.g. mastodon.social',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
search: [
|
|
|
|
{
|
|
|
|
text: 'Search term',
|
|
|
|
name: 'query',
|
|
|
|
type: 'text',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
'account-statuses': [
|
|
|
|
{
|
|
|
|
text: '@',
|
|
|
|
name: 'id',
|
|
|
|
type: 'text',
|
|
|
|
placeholder: 'cheeaun@mastodon.social',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
hashtag: [
|
|
|
|
{
|
|
|
|
text: '#',
|
|
|
|
name: 'hashtag',
|
|
|
|
type: 'text',
|
|
|
|
placeholder: 'e.g PixelArt',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
export const SHORTCUTS_META = {
|
|
|
|
following: {
|
2023-02-18 15:48:24 +03:00
|
|
|
title: 'Home / Following',
|
2023-02-18 16:37:34 +03:00
|
|
|
path: (_, index) => (index === 0 ? '/' : '/following'),
|
2023-02-16 12:51:54 +03:00
|
|
|
icon: 'home',
|
|
|
|
},
|
|
|
|
notifications: {
|
|
|
|
title: 'Notifications',
|
|
|
|
path: '/notifications',
|
|
|
|
icon: 'notification',
|
|
|
|
},
|
|
|
|
list: {
|
|
|
|
title: async ({ id }) => {
|
|
|
|
const list = await api().masto.v1.lists.fetch(id);
|
|
|
|
return list.title;
|
|
|
|
},
|
|
|
|
path: ({ id }) => `/l/${id}`,
|
|
|
|
icon: 'list',
|
|
|
|
},
|
|
|
|
public: {
|
|
|
|
title: ({ local, instance }) =>
|
|
|
|
`${local ? 'Local' : 'Federated'} (${instance})`,
|
|
|
|
path: ({ local, instance }) => `/${instance}/p${local ? '/l' : ''}`,
|
|
|
|
icon: ({ local }) => (local ? 'group' : 'earth'),
|
|
|
|
},
|
|
|
|
search: {
|
|
|
|
title: ({ query }) => query,
|
|
|
|
path: ({ query }) => `/search?q=${query}`,
|
|
|
|
icon: 'search',
|
|
|
|
},
|
|
|
|
'account-statuses': {
|
|
|
|
title: async ({ id }) => {
|
|
|
|
const account = await api().masto.v1.accounts.fetch(id);
|
|
|
|
return account.username || account.acct || account.displayName;
|
|
|
|
},
|
|
|
|
path: ({ id }) => `/a/${id}`,
|
|
|
|
icon: 'user',
|
|
|
|
},
|
|
|
|
bookmarks: {
|
|
|
|
title: 'Bookmarks',
|
|
|
|
path: '/b',
|
|
|
|
icon: 'bookmark',
|
|
|
|
},
|
|
|
|
favourites: {
|
|
|
|
title: 'Favourites',
|
|
|
|
path: '/f',
|
|
|
|
icon: 'heart',
|
|
|
|
},
|
|
|
|
hashtag: {
|
|
|
|
title: ({ hashtag }) => hashtag,
|
|
|
|
path: ({ hashtag }) => `/t/${hashtag}`,
|
|
|
|
icon: 'hashtag',
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
function ShortcutsSettings() {
|
|
|
|
const snapStates = useSnapshot(states);
|
|
|
|
const { masto } = api();
|
2023-02-18 14:39:17 +03:00
|
|
|
const { shortcuts } = snapStates;
|
2023-02-16 12:51:54 +03:00
|
|
|
|
|
|
|
const [lists, setLists] = useState([]);
|
|
|
|
const [followedHashtags, setFollowedHashtags] = useState([]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
const lists = await masto.v1.lists.list();
|
|
|
|
setLists(lists);
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
|
|
|
|
(async () => {
|
|
|
|
try {
|
|
|
|
const iterator = masto.v1.followedTags.list();
|
|
|
|
const tags = [];
|
|
|
|
do {
|
|
|
|
const { value, done } = await iterator.next();
|
|
|
|
if (done || value?.length === 0) break;
|
|
|
|
tags.push(...value);
|
|
|
|
} while (true);
|
|
|
|
setFollowedHashtags(tags);
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
})();
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div id="shortcuts-settings-container" class="sheet" tabindex="-1">
|
|
|
|
<header>
|
|
|
|
<h2>
|
|
|
|
<Icon icon="shortcut" /> Shortcuts{' '}
|
|
|
|
<sup
|
|
|
|
style={{
|
|
|
|
fontSize: 12,
|
|
|
|
opacity: 0.5,
|
|
|
|
textTransform: 'uppercase',
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
beta
|
|
|
|
</sup>
|
|
|
|
</h2>
|
|
|
|
</header>
|
|
|
|
<main>
|
|
|
|
<p>
|
|
|
|
Specify a list of shortcuts that'll appear in the floating Shortcuts
|
|
|
|
button.
|
|
|
|
</p>
|
2023-02-18 15:48:24 +03:00
|
|
|
<p>
|
|
|
|
<details>
|
|
|
|
<summary class="insignificant">
|
|
|
|
Experimental Multi-column mode
|
|
|
|
</summary>
|
|
|
|
<label>
|
|
|
|
<input
|
|
|
|
type="checkbox"
|
|
|
|
checked={snapStates.settings.shortcutsColumnsMode}
|
|
|
|
onChange={(e) => {
|
|
|
|
states.settings.shortcutsColumnsMode = e.target.checked;
|
|
|
|
}}
|
|
|
|
/>{' '}
|
|
|
|
Show shortcuts in multiple columns instead of the floating button.
|
|
|
|
</label>
|
|
|
|
</details>
|
|
|
|
</p>
|
|
|
|
{shortcuts.length > 0 ? (
|
2023-02-16 12:51:54 +03:00
|
|
|
<ol class="shortcuts-list">
|
2023-02-18 14:39:17 +03:00
|
|
|
{shortcuts.map((shortcut, i) => {
|
2023-02-16 12:51:54 +03:00
|
|
|
const key = i + Object.values(shortcut);
|
|
|
|
const { type } = shortcut;
|
2023-02-19 05:42:56 +03:00
|
|
|
if (!SHORTCUTS_META[type]) return null;
|
2023-02-16 12:51:54 +03:00
|
|
|
let { icon, title } = SHORTCUTS_META[type];
|
|
|
|
if (typeof title === 'function') {
|
|
|
|
title = title(shortcut);
|
|
|
|
}
|
|
|
|
if (typeof icon === 'function') {
|
|
|
|
icon = icon(shortcut);
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
<li key={key}>
|
|
|
|
<Icon icon={icon} />
|
|
|
|
<span class="shortcut-text">
|
|
|
|
<AsyncText>{title}</AsyncText>
|
|
|
|
</span>
|
|
|
|
<span>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="plain small"
|
|
|
|
disabled={i === 0}
|
|
|
|
onClick={() => {
|
|
|
|
const shortcutsArr = Array.from(states.shortcuts);
|
|
|
|
if (i > 0) {
|
|
|
|
const temp = states.shortcuts[i - 1];
|
|
|
|
shortcutsArr[i - 1] = shortcut;
|
|
|
|
shortcutsArr[i] = temp;
|
|
|
|
states.shortcuts = shortcutsArr;
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Icon icon="arrow-up" alt="Move up" />
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="plain small"
|
2023-02-18 14:39:17 +03:00
|
|
|
disabled={i === shortcuts.length - 1}
|
2023-02-16 12:51:54 +03:00
|
|
|
onClick={() => {
|
|
|
|
const shortcutsArr = Array.from(states.shortcuts);
|
|
|
|
if (i < states.shortcuts.length - 1) {
|
|
|
|
const temp = states.shortcuts[i + 1];
|
|
|
|
shortcutsArr[i + 1] = shortcut;
|
|
|
|
shortcutsArr[i] = temp;
|
|
|
|
states.shortcuts = shortcutsArr;
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Icon icon="arrow-down" alt="Move down" />
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="plain small"
|
|
|
|
onClick={() => {
|
|
|
|
states.shortcuts.splice(i, 1);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Icon icon="x" alt="Remove" />
|
|
|
|
</button>
|
|
|
|
</span>
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</ol>
|
|
|
|
) : (
|
|
|
|
<p class="ui-state insignificant">
|
|
|
|
No shortcuts yet. Add one from the form below.
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
<hr />
|
|
|
|
<ShortcutForm
|
2023-02-18 14:39:17 +03:00
|
|
|
disabled={shortcuts.length >= SHORTCUTS_LIMIT}
|
2023-02-16 12:51:54 +03:00
|
|
|
lists={lists}
|
|
|
|
followedHashtags={followedHashtags}
|
|
|
|
onSubmit={(data) => {
|
|
|
|
console.log('onSubmit', data);
|
|
|
|
states.shortcuts.push(data);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</main>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ShortcutsSettings;
|
2023-02-18 14:39:17 +03:00
|
|
|
function ShortcutForm({ type, lists, followedHashtags, onSubmit, disabled }) {
|
2023-02-16 12:51:54 +03:00
|
|
|
const [currentType, setCurrentType] = useState(type);
|
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<form
|
|
|
|
onSubmit={(e) => {
|
|
|
|
// Construct a nice object from form
|
|
|
|
e.preventDefault();
|
|
|
|
const data = new FormData(e.target);
|
|
|
|
const result = {};
|
|
|
|
data.forEach((value, key) => {
|
|
|
|
result[key] = value;
|
|
|
|
});
|
|
|
|
if (!result.type) return;
|
|
|
|
onSubmit(result);
|
|
|
|
// Reset
|
|
|
|
e.target.reset();
|
|
|
|
setCurrentType(null);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<header>
|
|
|
|
<h3>Add a shortcut</h3>
|
2023-02-18 14:39:17 +03:00
|
|
|
<button type="submit" disabled={disabled}>
|
|
|
|
Add
|
|
|
|
</button>
|
2023-02-16 12:51:54 +03:00
|
|
|
</header>
|
|
|
|
<p>
|
|
|
|
<label>
|
|
|
|
<span>Timeline</span>
|
|
|
|
<select
|
2023-02-18 14:39:17 +03:00
|
|
|
required
|
|
|
|
disabled={disabled}
|
2023-02-16 12:51:54 +03:00
|
|
|
onChange={(e) => {
|
|
|
|
setCurrentType(e.target.value);
|
|
|
|
}}
|
|
|
|
name="type"
|
|
|
|
>
|
|
|
|
<option></option>
|
|
|
|
{TYPES.map((type) => (
|
|
|
|
<option value={type}>{TYPE_TEXT[type]}</option>
|
|
|
|
))}
|
|
|
|
</select>
|
|
|
|
</label>
|
|
|
|
</p>
|
|
|
|
{TYPE_PARAMS[currentType]?.map?.(
|
|
|
|
({ text, name, type, placeholder }) => {
|
|
|
|
if (currentType === 'list') {
|
|
|
|
return (
|
|
|
|
<p>
|
|
|
|
<label>
|
|
|
|
<span>List</span>
|
2023-02-18 14:39:17 +03:00
|
|
|
<select name="id" required disabled={disabled}>
|
2023-02-16 12:51:54 +03:00
|
|
|
{lists.map((list) => (
|
|
|
|
<option value={list.id}>{list.title}</option>
|
|
|
|
))}
|
|
|
|
</select>
|
|
|
|
</label>
|
|
|
|
</p>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<p>
|
|
|
|
<label>
|
|
|
|
<span>{text}</span>{' '}
|
2023-02-16 13:22:19 +03:00
|
|
|
<input
|
|
|
|
type={type}
|
|
|
|
name={name}
|
|
|
|
placeholder={placeholder}
|
|
|
|
required={type === 'text'}
|
2023-02-18 14:39:17 +03:00
|
|
|
disabled={disabled}
|
2023-02-16 13:37:57 +03:00
|
|
|
list={
|
|
|
|
currentType === 'hashtag'
|
|
|
|
? 'followed-hashtags-datalist'
|
|
|
|
: null
|
|
|
|
}
|
2023-02-16 13:22:19 +03:00
|
|
|
/>
|
2023-02-16 12:51:54 +03:00
|
|
|
{currentType === 'hashtag' && followedHashtags.length > 0 && (
|
2023-02-16 13:37:57 +03:00
|
|
|
<datalist id="followed-hashtags-datalist">
|
2023-02-16 12:51:54 +03:00
|
|
|
{followedHashtags.map((tag) => (
|
|
|
|
<option value={tag.name} />
|
|
|
|
))}
|
|
|
|
</datalist>
|
|
|
|
)}
|
|
|
|
</label>
|
|
|
|
</p>
|
|
|
|
);
|
|
|
|
},
|
|
|
|
)}
|
|
|
|
</form>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
}
|