import './login.css'; import { useEffect, useRef, useState } from 'preact/hooks'; import { useSearchParams } from 'react-router-dom'; import Link from '../components/link'; import Loader from '../components/loader'; import instancesListURL from '../data/instances.json?url'; import { getAuthorizationURL, registerApplication } from '../utils/auth'; import store from '../utils/store'; import useTitle from '../utils/useTitle'; function Login() { 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'); const [instanceText, setInstanceText] = useState( instance || cachedInstanceURL?.toLowerCase() || '', ); 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); } })(); }, []); // useEffect(() => { // if (cachedInstanceURL) { // instanceURLRef.current.value = cachedInstanceURL.toLowerCase(); // } // }, []); const submitInstance = (instanceURL) => { store.local.set('instanceURL', instanceURL); (async () => { setUIState('loading'); try { const { client_id, client_secret, vapid_key } = await registerApplication({ instanceURL, }); if (client_id && client_secret) { store.session.set('clientID', client_id); store.session.set('clientSecret', client_secret); store.session.set('vapidKey', vapid_key); location.href = await getAuthorizationURL({ instanceURL, client_id, }); } else { alert('Failed to register application'); } setUIState('default'); } catch (e) { console.error(e); setUIState('error'); } })(); }; 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) : []; return (

Log in

{uiState === 'error' && (

Failed to log in. Please try again or another instance.

)}
{' '}
); } export default Login;