phanpy/src/pages/login.jsx

184 lines
5.3 KiB
React
Raw Normal View History

2022-12-21 01:04:04 +08:00
import './login.css';
2022-12-10 17:14:48 +08:00
import { useEffect, useRef, useState } from 'preact/hooks';
import { useSearchParams } from 'react-router-dom';
2022-12-10 17:14:48 +08:00
import Link from '../components/link';
2022-12-10 17:14:48 +08:00
import Loader from '../components/loader';
import instancesListURL from '../data/instances.json?url';
2022-12-10 17:14:48 +08:00
import { getAuthorizationURL, registerApplication } from '../utils/auth';
import store from '../utils/store';
import useTitle from '../utils/useTitle';
2022-12-16 13:27:04 +08:00
function Login() {
2022-12-10 17:14:48 +08:00
useTitle('Log in');
const instanceURLRef = useRef();
const cachedInstanceURL = store.local.get('instanceURL');
const [uiState, setUIState] = useState('default');
const [searchParams] = useSearchParams();
const instance = searchParams.get('instance');
2023-04-18 23:33:59 +08:00
const [instanceText, setInstanceText] = useState(
instance || cachedInstanceURL?.toLowerCase() || '',
2023-04-18 23:33:59 +08:00
);
2022-12-10 17:14:48 +08:00
const [instancesList, setInstancesList] = useState([]);
useEffect(() => {
(async () => {
try {
const res = await fetch(instancesListURL);
const data = await res.json();
setInstancesList(data);
} catch (e) {
// Silently fail
console.error(e);
}
})();
}, []);
2023-04-18 23:33:59 +08:00
// useEffect(() => {
// if (cachedInstanceURL) {
// instanceURLRef.current.value = cachedInstanceURL.toLowerCase();
// }
// }, []);
2022-12-10 17:14:48 +08:00
2023-04-18 23:33:59 +08:00
const submitInstance = (instanceURL) => {
2022-12-10 17:14:48 +08:00
store.local.set('instanceURL', instanceURL);
(async () => {
setUIState('loading');
try {
const { client_id, client_secret, vapid_key } =
await registerApplication({
instanceURL,
});
2022-12-10 17:14:48 +08:00
if (client_id && client_secret) {
store.session.set('clientID', client_id);
store.session.set('clientSecret', client_secret);
store.session.set('vapidKey', vapid_key);
2022-12-10 17:14:48 +08:00
location.href = await getAuthorizationURL({
instanceURL,
client_id,
});
} else {
alert('Failed to register application');
}
setUIState('default');
} catch (e) {
console.error(e);
setUIState('error');
}
})();
};
2023-04-18 23:33:59 +08:00
const onSubmit = (e) => {
e.preventDefault();
const { elements } = e.target;
let instanceURL = elements.instanceURL.value.toLowerCase();
// Remove protocol from instance URL
instanceURL = instanceURL.replace(/^https?:\/\//, '').replace(/\/+$/, '');
// Remove @acct@ or acct@ from instance URL
instanceURL = instanceURL.replace(/^@?[^@]+@/, '');
if (!/\./.test(instanceURL)) {
instanceURL = instancesList.find((instance) =>
instance.includes(instanceURL),
);
}
submitInstance(instanceURL);
};
const instancesSuggestions = instanceText
? instancesList
.filter((instance) => instance.includes(instanceText))
.sort((a, b) => {
// Move text that starts with instanceText to the start
const aStartsWith = a
.toLowerCase()
.startsWith(instanceText.toLowerCase());
const bStartsWith = b
.toLowerCase()
.startsWith(instanceText.toLowerCase());
if (aStartsWith && !bStartsWith) return -1;
if (!aStartsWith && bStartsWith) return 1;
return 0;
})
.slice(0, 10)
: [];
2022-12-10 17:14:48 +08:00
return (
2022-12-21 01:04:04 +08:00
<main id="login" style={{ textAlign: 'center' }}>
2022-12-10 17:14:48 +08:00
<form onSubmit={onSubmit}>
<h1>Log in</h1>
<label>
<p>Instance</p>
<input
2023-04-18 23:33:59 +08:00
value={instanceText}
2022-12-10 17:14:48 +08:00
required
type="text"
class="large"
id="instanceURL"
ref={instanceURLRef}
disabled={uiState === 'loading'}
2023-04-18 23:33:59 +08:00
// list="instances-list"
autocorrect="off"
autocapitalize="off"
autocomplete="off"
spellcheck={false}
2023-04-18 23:33:59 +08:00
placeholder="instance domain"
onInput={(e) => {
setInstanceText(e.target.value);
}}
2022-12-10 17:14:48 +08:00
/>
{instancesSuggestions?.length > 0 ? (
<ul id="instances-suggestions">
{instancesSuggestions.map((instance) => (
2023-04-18 23:33:59 +08:00
<li>
<button
type="button"
class="plain4"
onClick={() => {
submitInstance(instance);
}}
>
{instance}
</button>
</li>
))}
</ul>
) : (
<div id="instances-eg">e.g. &ldquo;mastodon.social&rsquo;</div>
)}
2023-04-18 23:33:59 +08:00
{/* <datalist id="instances-list">
2022-12-10 17:14:48 +08:00
{instancesList.map((instance) => (
<option value={instance} />
))}
2023-04-18 23:33:59 +08:00
</datalist> */}
2022-12-10 17:14:48 +08:00
</label>
{uiState === 'error' && (
<p class="error">
Failed to log in. Please try again or another instance.
</p>
)}
2023-04-18 23:33:59 +08:00
<div>
2022-12-10 17:14:48 +08:00
<button class="large" disabled={uiState === 'loading'}>
Log in
</button>{' '}
2023-04-18 23:33:59 +08:00
</div>
2022-12-21 01:04:04 +08:00
<Loader hidden={uiState !== 'loading'} />
2022-12-10 17:14:48 +08:00
<hr />
<p>
<a href="https://joinmastodon.org/servers" target="_blank">
Don't have an account? Create one!
</a>
</p>
<p>
<Link to="/">Go home</Link>
2022-12-10 17:14:48 +08:00
</p>
</form>
</main>
);
2022-12-16 13:27:04 +08:00
}
export default Login;