phanpy/src/pages/login.jsx

228 lines
6.7 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
2023-12-10 19:16:34 +08:00
import logo from '../assets/logo.svg';
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';
2023-12-25 19:25:48 +08:00
const { PHANPY_DEFAULT_INSTANCE: DEFAULT_INSTANCE } = import.meta.env;
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-12-25 19:25:48 +08:00
const submit = searchParams.get('submit');
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) => {
2023-12-10 19:16:34 +08:00
if (!instanceURL) return;
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-12-10 19:16:34 +08:00
const cleanInstanceText = instanceText
? instanceText
.replace(/^https?:\/\//, '') // Remove protocol from instance URL
.replace(/\/+$/, '') // Remove trailing slash
.replace(/^@?[^@]+@/, '') // Remove @?acct@
.trim()
: null;
const instanceTextLooksLikeDomain =
/[^\s\r\n\t\/\\]+\.[^\s\r\n\t\/\\]+/.test(cleanInstanceText) &&
!/[\s\/\\@]/.test(cleanInstanceText);
2023-04-18 23:33:59 +08:00
2023-12-10 19:16:34 +08:00
const instancesSuggestions = cleanInstanceText
? 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)
: [];
2023-12-10 19:16:34 +08:00
const selectedInstanceText = instanceTextLooksLikeDomain
? cleanInstanceText
: instancesSuggestions?.length
? instancesSuggestions[0]
: instanceText
? instancesList.find((instance) => instance.includes(instanceText))
: null;
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);
submitInstance(selectedInstanceText);
};
2023-12-25 19:25:48 +08:00
if (submit) {
useEffect(() => {
submitInstance(instance || selectedInstanceText);
}, []);
}
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}>
2023-12-10 19:16:34 +08:00
<h1>
<img src={logo} alt="" width="80" height="80" />
<br />
Log in
</h1>
2022-12-10 17:14:48 +08:00
<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">
2023-12-10 19:16:34 +08:00
{instancesSuggestions.map((instance, i) => (
2023-04-18 23:33:59 +08:00
<li>
<button
type="button"
2023-12-10 19:16:34 +08:00
class="plain5"
2023-04-18 23:33:59 +08:00
onClick={() => {
submitInstance(instance);
}}
>
{instance}
</button>
</li>
))}
</ul>
) : (
2023-12-10 19:16:34 +08:00
<div id="instances-eg">e.g. &ldquo;mastodon.social&rdquo;</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>
2023-12-10 19:16:34 +08:00
<button
disabled={
uiState === 'loading' || !instanceText || !selectedInstanceText
}
>
{selectedInstanceText
? `Continue with ${selectedInstanceText}`
: 'Continue'}
2022-12-10 17:14:48 +08:00
</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 />
2023-12-25 19:25:48 +08:00
{!DEFAULT_INSTANCE && (
<p>
<a href="https://joinmastodon.org/servers" target="_blank">
Don't have an account? Create one!
</a>
</p>
)}
2022-12-10 17:14:48 +08:00
<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;