forked from Mirrors/elk
feat: wip
This commit is contained in:
parent
f7df0e54f5
commit
313cafa23c
18 changed files with 146 additions and 129 deletions
4
app.vue
4
app.vue
|
@ -3,9 +3,7 @@ setupPageHeader()
|
||||||
provideGlobalCommands()
|
provideGlobalCommands()
|
||||||
|
|
||||||
// We want to trigger rerendering the page when account changes
|
// We want to trigger rerendering the page when account changes
|
||||||
const key = computed(() =>
|
const key = computed(() => currentUser.value ? getUniqueUserId(currentUser.value) : '')
|
||||||
`${currentServer.value}:${checkUser(currentUser.value) ? currentUser.value.account.id : GUEST_ID}`,
|
|
||||||
)
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
|
|
18
components/account/AccountGuest.vue
Normal file
18
components/account/AccountGuest.vue
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import type { UserLogin } from '~/types'
|
||||||
|
|
||||||
|
defineProps<{
|
||||||
|
user: UserLogin
|
||||||
|
}>()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div flex="~ gap3" items-center>
|
||||||
|
<div bg="gray/40" rounded-full w-54px h-54px flex shrink-0 items-center justify-center text-5>
|
||||||
|
G
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
Guest from <span font-bold>{{ user.server }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
|
@ -7,7 +7,7 @@ const { account } = defineProps<{
|
||||||
}>()
|
}>()
|
||||||
let relationship = $(useRelationship(account))
|
let relationship = $(useRelationship(account))
|
||||||
|
|
||||||
const isSelf = $computed(() => currentUser.value?.account.id === account.id)
|
const isSelf = $computed(() => checkUser(currentUser.value) && currentUser.value.account.id === account.id)
|
||||||
|
|
||||||
const masto = useMasto()
|
const masto = useMasto()
|
||||||
const toggleMute = async () => {
|
const toggleMute = async () => {
|
||||||
|
|
|
@ -9,8 +9,8 @@
|
||||||
w-8
|
w-8
|
||||||
:draggable="false"
|
:draggable="false"
|
||||||
/>
|
/>
|
||||||
<div v-else>
|
<div v-else bg="gray/40" rounded-full w-8 h-8 flex items-center justify-center text-5>
|
||||||
TODO: Guest
|
G
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
const paginator = useMasto().accounts.iterateStatuses(currentUser.value!.account.id, { pinned: true })
|
const paginator = useMasto().accounts.iterateStatuses(currentUser.value!.account!.id, { pinned: true })
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
|
|
|
@ -16,7 +16,9 @@ const masto = useMasto()
|
||||||
@click="switchUser(user, masto)"
|
@click="switchUser(user, masto)"
|
||||||
>
|
>
|
||||||
<AccountAvatar v-if="!user.guest" w-13 h-13 :account="user.account" />
|
<AccountAvatar v-if="!user.guest" w-13 h-13 :account="user.account" />
|
||||||
<span v-else>TODO: Guest from {{ user.server }}</span>
|
<div v-else bg="gray/40" rounded-full w-13 h-13 flex shrink-0 items-center justify-center text-5>
|
||||||
|
G
|
||||||
|
</div>
|
||||||
</button>
|
</button>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -48,7 +48,7 @@ async function oauth() {
|
||||||
}
|
}
|
||||||
|
|
||||||
function explore() {
|
function explore() {
|
||||||
masto.loginTo({ server, guest: true })
|
masto.loginTo({ server, guestOnly: true })
|
||||||
}
|
}
|
||||||
|
|
||||||
async function handleInput() {
|
async function handleInput() {
|
||||||
|
|
|
@ -22,7 +22,7 @@ const masto = useMasto()
|
||||||
@click="switchUser(user, masto)"
|
@click="switchUser(user, masto)"
|
||||||
>
|
>
|
||||||
<AccountInfo v-if="!user.guest" :account="user.account" :hover-card="false" />
|
<AccountInfo v-if="!user.guest" :account="user.account" :hover-card="false" />
|
||||||
<span v-else>TODO: Guest from {{ user.server }}</span>
|
<AccountGuest v-else :user="user" />
|
||||||
<div flex-auto />
|
<div flex-auto />
|
||||||
<div v-if="isSameUser(user, currentUser)" i-ri:check-line text-primary mya text-2xl />
|
<div v-if="isSameUser(user, currentUser)" i-ri:check-line text-primary mya text-2xl />
|
||||||
</button>
|
</button>
|
||||||
|
@ -34,7 +34,7 @@ const masto = useMasto()
|
||||||
@click="openSigninDialog"
|
@click="openSigninDialog"
|
||||||
/>
|
/>
|
||||||
<CommonDropdownItem
|
<CommonDropdownItem
|
||||||
v-if="isMastoInitialised && currentUser"
|
v-if="isMastoInitialised && canSignOut"
|
||||||
:text="$t('user.sign_out_account', [getFullHandle(currentUser)])"
|
:text="$t('user.sign_out_account', [getFullHandle(currentUser)])"
|
||||||
icon="i-ri:logout-box-line rtl-flip"
|
icon="i-ri:logout-box-line rtl-flip"
|
||||||
@click="signout"
|
@click="signout"
|
||||||
|
|
|
@ -248,7 +248,7 @@ export const provideGlobalCommands = () => {
|
||||||
useCommand({
|
useCommand({
|
||||||
scope: 'Actions',
|
scope: 'Actions',
|
||||||
|
|
||||||
visible: () => currentUser.value,
|
visible: () => !isGuest.value,
|
||||||
|
|
||||||
name: () => t('action.compose'),
|
name: () => t('action.compose'),
|
||||||
icon: 'i-ri:quill-pen-line',
|
icon: 'i-ri:quill-pen-line',
|
||||||
|
@ -344,9 +344,9 @@ export const provideGlobalCommands = () => {
|
||||||
parent: 'account-switch',
|
parent: 'account-switch',
|
||||||
scope: 'Switch account',
|
scope: 'Switch account',
|
||||||
|
|
||||||
visible: () => !user.guest && user.account.id !== currentUser.value?.account?.id,
|
visible: () => !isSameUser(user, currentUser.value),
|
||||||
|
|
||||||
name: () => t('command.switch_account', [getFullHandle(user)]),
|
name: () => t('command.switch_account', [user.guest ? `guest from ${user.server}` : getFullHandle(user)]),
|
||||||
icon: 'i-ri:user-shared-line',
|
icon: 'i-ri:user-shared-line',
|
||||||
|
|
||||||
onActivate() {
|
onActivate() {
|
||||||
|
@ -356,8 +356,8 @@ export const provideGlobalCommands = () => {
|
||||||
useCommand({
|
useCommand({
|
||||||
scope: 'Account',
|
scope: 'Account',
|
||||||
|
|
||||||
visible: () => currentUser.value,
|
visible: () => canSignOut.value,
|
||||||
name: () => currentUser.value ? t('user.sign_out_account', [getFullHandle(currentUser.value)]) : '',
|
name: () => t('user.sign_out_account', [getFullHandle(currentUser.value)]),
|
||||||
icon: 'i-ri:logout-box-line',
|
icon: 'i-ri:logout-box-line',
|
||||||
|
|
||||||
onActivate() {
|
onActivate() {
|
||||||
|
|
|
@ -97,7 +97,7 @@ async function subscribe(
|
||||||
|
|
||||||
async function unsubscribeFromBackend(fromSWPushManager: boolean, removePushNotification = true) {
|
async function unsubscribeFromBackend(fromSWPushManager: boolean, removePushNotification = true) {
|
||||||
const cu = currentUser.value
|
const cu = currentUser.value
|
||||||
if (cu) {
|
if (checkUser(cu)) {
|
||||||
await removePushNotifications(cu)
|
await removePushNotifications(cu)
|
||||||
removePushNotification && await removePushNotificationData(cu, fromSWPushManager)
|
removePushNotification && await removePushNotificationData(cu, fromSWPushManager)
|
||||||
}
|
}
|
||||||
|
@ -105,7 +105,7 @@ async function unsubscribeFromBackend(fromSWPushManager: boolean, removePushNoti
|
||||||
|
|
||||||
async function removePushNotificationDataOnError(e: Error) {
|
async function removePushNotificationDataOnError(e: Error) {
|
||||||
const cu = currentUser.value
|
const cu = currentUser.value
|
||||||
if (cu)
|
if (checkUser(cu))
|
||||||
await removePushNotificationData(cu, true)
|
await removePushNotificationData(cu, true)
|
||||||
|
|
||||||
throw e
|
throw e
|
||||||
|
|
|
@ -117,6 +117,9 @@ export const usePushManager = () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
const saveSettings = async (policy?: SubscriptionPolicy) => {
|
const saveSettings = async (policy?: SubscriptionPolicy) => {
|
||||||
|
if (!checkUser(currentUser.value))
|
||||||
|
return
|
||||||
|
|
||||||
if (policy)
|
if (policy)
|
||||||
pushNotificationData.value.policy = policy
|
pushNotificationData.value.policy = policy
|
||||||
|
|
||||||
|
@ -133,6 +136,9 @@ export const usePushManager = () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
const undoChanges = () => {
|
const undoChanges = () => {
|
||||||
|
if (!checkUser(currentUser.value))
|
||||||
|
return
|
||||||
|
|
||||||
const current = pushNotificationData.value
|
const current = pushNotificationData.value
|
||||||
const previous = history.value[0].snapshot
|
const previous = history.value[0].snapshot
|
||||||
current.favourite = previous.favourite
|
current.favourite = previous.favourite
|
||||||
|
|
|
@ -51,7 +51,7 @@ function mentionHTML(acct: string) {
|
||||||
|
|
||||||
export function getReplyDraft(status: Status) {
|
export function getReplyDraft(status: Status) {
|
||||||
const accountsToMention: string[] = []
|
const accountsToMention: string[] = []
|
||||||
const userId = currentUser.value?.account.id
|
const userId = currentUser.value!.account!.id
|
||||||
if (status.account.id !== userId)
|
if (status.account.id !== userId)
|
||||||
accountsToMention.push(status.account.acct)
|
accountsToMention.push(status.account.acct)
|
||||||
accountsToMention.push(...(status.mentions.filter(mention => mention.id !== userId).map(mention => mention.acct)))
|
accountsToMention.push(...(status.mentions.filter(mention => mention.id !== userId).map(mention => mention.acct)))
|
||||||
|
|
|
@ -45,107 +45,133 @@ const users = await initializeUsers()
|
||||||
const instances = useLocalStorage<Record<string, Instance>>(STORAGE_KEY_SERVERS, mock ? mock.server : {}, { deep: true })
|
const instances = useLocalStorage<Record<string, Instance>>(STORAGE_KEY_SERVERS, mock ? mock.server : {}, { deep: true })
|
||||||
const currentUserId = useLocalStorage<string>(STORAGE_KEY_CURRENT_USER, mock ? mock.user.account.id : '')
|
const currentUserId = useLocalStorage<string>(STORAGE_KEY_CURRENT_USER, mock ? mock.user.account.id : '')
|
||||||
const isGuestId = computed(() => !currentUserId.value || currentUserId.value.startsWith(`${GUEST_ID}@`))
|
const isGuestId = computed(() => !currentUserId.value || currentUserId.value.startsWith(`${GUEST_ID}@`))
|
||||||
|
const defaultUser: UserLogin<false> = {
|
||||||
|
server: DEFAULT_SERVER,
|
||||||
|
guest: true,
|
||||||
|
}
|
||||||
|
|
||||||
export const currentUser = computed<UserLogin | undefined>(() => {
|
export const currentUser = computed<UserLogin>(() => {
|
||||||
if (!currentUserId.value)
|
let user: UserLogin | undefined
|
||||||
// Fallback to the first account
|
if (!currentUserId.value) {
|
||||||
return users.value[0]
|
// Fallback to the first account
|
||||||
|
user = users.value[0]
|
||||||
if (isGuestId.value) {
|
|
||||||
const server = currentUserId.value.replace(`${GUEST_ID}@`, '')
|
|
||||||
return users.value.find(user => user.guest && user.server === server)
|
|
||||||
}
|
}
|
||||||
|
else if (isGuestId.value) {
|
||||||
return users.value.find(user => user.account?.id === currentUserId.value)
|
const server = currentUserId.value.replace(`${GUEST_ID}@`, '')
|
||||||
|
user = users.value.find(user => user.guest && user.server === server)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
user = users.value.find(user => user.account?.id === currentUserId.value)
|
||||||
|
}
|
||||||
|
return user || defaultUser
|
||||||
})
|
})
|
||||||
|
|
||||||
export const currentServer = computed<string>(() => currentUser.value?.server || DEFAULT_SERVER)
|
export const currentServer = computed<string>(() => currentUser.value.server)
|
||||||
export const currentInstance = computed<null | Instance>(() => {
|
export const currentInstance = computed<null | Instance>(() => {
|
||||||
return instances.value[currentServer.value] ?? null
|
return instances.value[currentServer.value] ?? null
|
||||||
})
|
})
|
||||||
export const checkUser = (val: UserLogin | undefined): val is UserLogin<true> => !!(val && !val.guest)
|
export const checkUser = (val: UserLogin | undefined): val is UserLogin<true> => !!(val && !val.guest)
|
||||||
export const isGuest = computed(() => !checkUser(currentUser.value))
|
export const isGuest = computed(() => !checkUser(currentUser.value))
|
||||||
export const getUniqueUserId = (user: UserLogin) => user.guest ? `${GUEST_ID}@${user.server}` : user.account.id
|
export const getUniqueUserId = (user: UserLogin) =>
|
||||||
|
user.guest ? `${GUEST_ID}@${user.server}` : user.account.id
|
||||||
export const isSameUser = (a: UserLogin | undefined, b: UserLogin | undefined) =>
|
export const isSameUser = (a: UserLogin | undefined, b: UserLogin | undefined) =>
|
||||||
a && b && getUniqueUserId(a) === getUniqueUserId(b)
|
a && b && getUniqueUserId(a) === getUniqueUserId(b)
|
||||||
|
|
||||||
export const currentUserHandle = computed(() =>
|
export const currentUserHandle = computed(() =>
|
||||||
isGuestId.value ? GUEST_ID : currentUser.value!.account!.acct
|
currentUser.value.guest ? GUEST_ID : currentUser.value.account!.acct,
|
||||||
,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
export const useUsers = () => users
|
export const useUsers = () => users
|
||||||
|
|
||||||
export const characterLimit = computed(() => currentInstance.value?.configuration.statuses.maxCharacters ?? DEFAULT_POST_CHARS_LIMIT)
|
export const characterLimit = computed(() => currentInstance.value?.configuration.statuses.maxCharacters ?? DEFAULT_POST_CHARS_LIMIT)
|
||||||
|
|
||||||
async function loginTo(user?: UserLogin) {
|
async function loginTo({ server, token, vapidKey, pushSubscription, guest = false }: { guest?: boolean } & Omit<UserLogin, 'guest'>) {
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const server = user?.server || (route.params.server as string) || DEFAULT_SERVER
|
|
||||||
|
let user: UserLogin | undefined = token
|
||||||
|
? users.value.find(u => u.server === server && u.token === token)
|
||||||
|
: ((guest
|
||||||
|
? undefined
|
||||||
|
: users.value.find(u => u.server === server && u.token))
|
||||||
|
|| users.value.find(u => u.server === server && u.guest))
|
||||||
|
|
||||||
|
const needPush = !user
|
||||||
|
if (!user) {
|
||||||
|
if (token) {
|
||||||
|
user = {
|
||||||
|
server,
|
||||||
|
guest: false,
|
||||||
|
token,
|
||||||
|
vapidKey,
|
||||||
|
pushSubscription,
|
||||||
|
account: undefined as any, // to be assigned later
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
user = { server, guest: true }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const masto = await loginMasto({
|
const masto = await loginMasto({
|
||||||
url: `https://${server}`,
|
url: `https://${user.server}`,
|
||||||
accessToken: user?.token,
|
accessToken: user.token,
|
||||||
disableVersionCheck: true,
|
disableVersionCheck: true,
|
||||||
// Suppress warning of `masto/fetch` usage
|
// Suppress warning of `masto/fetch` usage
|
||||||
disableExperimentalWarning: true,
|
disableExperimentalWarning: true,
|
||||||
})
|
})
|
||||||
|
|
||||||
if (!user?.token) {
|
if (user.guest) {
|
||||||
const instance = await masto.instances.fetch()
|
const instance = await masto.instances.fetch()
|
||||||
|
|
||||||
instances.value[server] = instance
|
instances.value[server] = instance
|
||||||
if (!users.value.some(u => u.server === server && u.guest))
|
|
||||||
users.value.push({ server, guest: true })
|
|
||||||
|
|
||||||
currentUserId.value = `${GUEST_ID}@${server}`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
else {
|
else {
|
||||||
try {
|
const [me, instance, pushSubscription] = await Promise.all([
|
||||||
const [me, instance, pushSubscription] = await Promise.all([
|
masto.accounts.verifyCredentials(),
|
||||||
masto.accounts.verifyCredentials(),
|
masto.instances.fetch(),
|
||||||
masto.instances.fetch(),
|
// if PWA is not enabled, don't get push subscription
|
||||||
// if PWA is not enabled, don't get push subscription
|
useRuntimeConfig().public.pwaEnabled
|
||||||
useRuntimeConfig().public.pwaEnabled
|
// we get 404 response instead empty data
|
||||||
// we get 404 response instead empty data
|
? masto.pushSubscriptions.fetch().catch(() => Promise.resolve(undefined))
|
||||||
? masto.pushSubscriptions.fetch().catch(() => Promise.resolve(undefined))
|
: Promise.resolve(undefined),
|
||||||
: Promise.resolve(undefined),
|
])
|
||||||
])
|
|
||||||
|
|
||||||
if (!me.acct.includes('@'))
|
if (!me.acct.includes('@'))
|
||||||
me.acct = `${me.acct}@${instance.uri}`
|
me.acct = `${me.acct}@${instance.uri}`
|
||||||
|
|
||||||
user.account = me
|
user.account = me
|
||||||
user.pushSubscription = pushSubscription
|
user.pushSubscription = pushSubscription
|
||||||
currentUserId.value = me.id
|
instances.value[server] = instance
|
||||||
instances.value[server] = instance
|
|
||||||
|
|
||||||
if (!users.value.some(u => u.server === user.server && u.token === user.token))
|
|
||||||
users.value.push(user)
|
|
||||||
}
|
|
||||||
catch {
|
|
||||||
await signout()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (needPush)
|
||||||
|
users.value.push(user)
|
||||||
|
|
||||||
|
currentUserId.value = getUniqueUserId(user)
|
||||||
|
|
||||||
// This only cleans up the URL; page content should stay the same
|
// This only cleans up the URL; page content should stay the same
|
||||||
if (route.path === '/signin/callback') {
|
if (route.path === '/signin/callback') {
|
||||||
await router.push('/home')
|
await router.push('/home')
|
||||||
}
|
}
|
||||||
|
|
||||||
else if ('server' in route.params && user?.token && !useNuxtApp()._processingMiddleware) {
|
else if ('server' in route.params && user.server !== route.params.server) {
|
||||||
await router.push({
|
await router.push({
|
||||||
...route,
|
...route,
|
||||||
|
params: {
|
||||||
|
...route.params,
|
||||||
|
server: user.server,
|
||||||
|
},
|
||||||
force: true,
|
force: true,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
return masto
|
return masto
|
||||||
}
|
}
|
||||||
|
export type LoginTo = typeof loginTo
|
||||||
|
|
||||||
export const switchUser = (user: UserLogin, masto: ElkMasto) => {
|
export const switchUser = (user: UserLogin, masto: ElkMasto) => {
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
if (!user.guest && !isGuest.value && user.account.id === currentUser.value!.account!.id)
|
if (!user.guest && !isGuest.value && user.account.id === currentUser.value.account!.id)
|
||||||
router.push(getAccountRoute(user.account))
|
router.push(getAccountRoute(user.account))
|
||||||
else
|
else
|
||||||
masto.loginTo(user)
|
masto.loginTo(user)
|
||||||
|
@ -173,7 +199,7 @@ export function getUsersIndexByUserId(userId: string) {
|
||||||
return users.value.findIndex(u => u.account?.id === userId)
|
return users.value.findIndex(u => u.account?.id === userId)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function removePushNotificationData(user: UserLogin, fromSWPushManager = true) {
|
export async function removePushNotificationData(user: UserLogin<true>, fromSWPushManager = true) {
|
||||||
// clear push subscription
|
// clear push subscription
|
||||||
user.pushSubscription = undefined
|
user.pushSubscription = undefined
|
||||||
const { acct } = user.account!
|
const { acct } = user.account!
|
||||||
|
@ -212,13 +238,18 @@ export async function removePushNotifications(user: UserLogin<true>) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// do not sign out if there is only one guest user
|
||||||
|
export const canSignOut = computed(() =>
|
||||||
|
users.value.length > 1 || !users.value[0].guest,
|
||||||
|
)
|
||||||
|
|
||||||
export async function signout() {
|
export async function signout() {
|
||||||
// TODO: confirm
|
// TODO: confirm
|
||||||
if (!currentUser.value)
|
|
||||||
|
if (!canSignOut.value)
|
||||||
return
|
return
|
||||||
|
|
||||||
const index = users.value.findIndex(u => isSameUser(u, currentUser.value))
|
const index = users.value.findIndex(u => isSameUser(u, currentUser.value))
|
||||||
|
|
||||||
if (index !== -1) {
|
if (index !== -1) {
|
||||||
// Clear stale data
|
// Clear stale data
|
||||||
clearUserLocalStorage()
|
clearUserLocalStorage()
|
||||||
|
@ -235,11 +266,8 @@ export async function signout() {
|
||||||
users.value.splice(index, 1)
|
users.value.splice(index, 1)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set currentUserId to next user if available
|
// Set currentUserId to next user
|
||||||
currentUserId.value = users.value[0]?.account?.id
|
currentUserId.value = getUniqueUserId(users.value[0] ? users.value[0] : defaultUser)
|
||||||
|
|
||||||
if (!currentUserId.value)
|
|
||||||
await useRouter().push('/')
|
|
||||||
|
|
||||||
const masto = useMasto()
|
const masto = useMasto()
|
||||||
await masto.loginTo(currentUser.value)
|
await masto.loginTo(currentUser.value)
|
||||||
|
@ -306,7 +334,7 @@ export function useUserLocalStorage<T extends object>(key: string, initial: () =
|
||||||
const all = storages.get(key) as Ref<Record<string, T>>
|
const all = storages.get(key) as Ref<Record<string, T>>
|
||||||
|
|
||||||
return computed(() => {
|
return computed(() => {
|
||||||
const id = isGuestId.value
|
const id = currentUser.value.guest
|
||||||
? GUEST_ID
|
? GUEST_ID
|
||||||
: currentUser.value!.account!.acct
|
: currentUser.value!.account!.acct
|
||||||
all.value[id] = Object.assign(initial(), all.value[id] || {})
|
all.value[id] = Object.assign(initial(), all.value[id] || {})
|
||||||
|
@ -343,10 +371,11 @@ export const createMasto = () => {
|
||||||
|
|
||||||
if (key === 'loginTo') {
|
if (key === 'loginTo') {
|
||||||
return (...args: any[]): Promise<MastoClient> => {
|
return (...args: any[]): Promise<MastoClient> => {
|
||||||
return apiPromise.value = loginTo(...args).then((r) => {
|
return apiPromise.value = (loginTo as any)(...args).then((r: any) => {
|
||||||
api.value = r
|
api.value = r
|
||||||
return masto
|
return masto
|
||||||
}).catch(() => {
|
}).catch((err: any) => {
|
||||||
|
console.error(err)
|
||||||
// Show error page when Mastodon server is down
|
// Show error page when Mastodon server is down
|
||||||
throw createError({
|
throw createError({
|
||||||
fatal: true,
|
fatal: true,
|
||||||
|
|
|
@ -22,7 +22,7 @@ const reload = async () => {
|
||||||
try {
|
try {
|
||||||
if (!masto.loggedIn.value)
|
if (!masto.loggedIn.value)
|
||||||
await masto.loginTo(currentUser.value)
|
await masto.loginTo(currentUser.value)
|
||||||
clearError({ redirect: currentUser.value ? '/home' : `/${currentServer.value}/public` })
|
clearError({ redirect: !isGuest.value ? '/home' : `/${currentServer.value}/public` })
|
||||||
}
|
}
|
||||||
catch {
|
catch {
|
||||||
state.value = 'error'
|
state.value = 'error'
|
||||||
|
|
|
@ -32,9 +32,8 @@ const wideLayout = computed(() => route.meta.wideLayout ?? false)
|
||||||
>
|
>
|
||||||
<AccountInfo :account="currentUser.account" md:break-words />
|
<AccountInfo :account="currentUser.account" md:break-words />
|
||||||
</NuxtLink>
|
</NuxtLink>
|
||||||
<div v-else>
|
<AccountGuest v-else :user="currentUser" />
|
||||||
TODO: guest {{ currentUser.server }} @ default.vue
|
|
||||||
</div>
|
|
||||||
<UserDropdown />
|
<UserDropdown />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
export default defineNuxtRouteMiddleware(async (to, from) => {
|
export default defineNuxtRouteMiddleware(async (to) => {
|
||||||
if (process.server)
|
if (process.server)
|
||||||
return
|
return
|
||||||
|
|
||||||
|
@ -13,23 +13,6 @@ export default defineNuxtRouteMiddleware(async (to, from) => {
|
||||||
|
|
||||||
const user = currentUser.value
|
const user = currentUser.value
|
||||||
|
|
||||||
if (!user) {
|
|
||||||
if (from.params.server !== to.params.server) {
|
|
||||||
await masto.loginTo({
|
|
||||||
server: to.params.server as string,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// No need to additionally resolve an id if we're already logged in
|
|
||||||
if (user.server === to.params.server)
|
|
||||||
return
|
|
||||||
|
|
||||||
// Tags don't need to be redirected to a local id
|
|
||||||
if (to.params.tag)
|
|
||||||
return
|
|
||||||
|
|
||||||
// Handle redirecting to new permalink structure for users with old links
|
// Handle redirecting to new permalink structure for users with old links
|
||||||
if (!to.params.server) {
|
if (!to.params.server) {
|
||||||
return {
|
return {
|
||||||
|
@ -41,28 +24,11 @@ export default defineNuxtRouteMiddleware(async (to, from) => {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
// No need to additionally resolve an id if we're already logged in
|
||||||
// If we're already on an account page, we can search for this on the new instance
|
if (user.server === to.params.server)
|
||||||
if (to.params.account && to.name !== 'status' && to.params.account.includes('@')) {
|
return
|
||||||
const account = await fetchAccountByHandle(to.params.account as string)
|
|
||||||
if (account)
|
|
||||||
return getAccountRoute(account)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!masto.loggedIn.value)
|
masto.loginTo({
|
||||||
await masto.loginTo(currentUser.value)
|
server: to.params.server as string,
|
||||||
|
})
|
||||||
// If we're logged in, search for the local id the account or status corresponds to
|
|
||||||
const { value } = await masto.search({ q: `https:/${to.fullPath}`, resolve: true, limit: 1 }).next()
|
|
||||||
|
|
||||||
const { accounts, statuses } = value
|
|
||||||
if (statuses[0])
|
|
||||||
return getStatusRoute(statuses[0])
|
|
||||||
|
|
||||||
if (accounts[0])
|
|
||||||
return getAccountRoute(accounts[0])
|
|
||||||
}
|
|
||||||
catch {}
|
|
||||||
|
|
||||||
return '/home'
|
|
||||||
})
|
})
|
||||||
|
|
|
@ -71,9 +71,7 @@ async function importTokens() {
|
||||||
<div flex="~ col gap2">
|
<div flex="~ col gap2">
|
||||||
<div v-for="user of loggedInUsers" :key="getUniqueUserId(user)">
|
<div v-for="user of loggedInUsers" :key="getUniqueUserId(user)">
|
||||||
<AccountInfo v-if="!user.guest" :account="user.account" :hover-card="false" />
|
<AccountInfo v-if="!user.guest" :account="user.account" :hover-card="false" />
|
||||||
<div v-else>
|
<AccountGuest v-else :user="user" />
|
||||||
TODO: Guest @ settings/users/index.vue
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div my4 border="t base" />
|
<div my4 border="t base" />
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import type { Account, AccountCredentials, Attachment, CreateStatusParams, Emoji, Instance, MastoClient, Notification, PushSubscription, Status } from 'masto'
|
import type { Account, AccountCredentials, Attachment, CreateStatusParams, Emoji, Instance, MastoClient, Notification, PushSubscription, Status } from 'masto'
|
||||||
import type { Ref } from 'vue'
|
import type { Ref } from 'vue'
|
||||||
import type { MarkNonNullable, Mutable } from './utils'
|
import type { MarkNonNullable, Mutable } from './utils'
|
||||||
|
import type { LoginTo } from '~/composables/users'
|
||||||
|
|
||||||
export interface AppInfo {
|
export interface AppInfo {
|
||||||
id: string
|
id: string
|
||||||
|
@ -30,7 +31,7 @@ export type UserLogin<WithToken extends boolean = boolean> = {
|
||||||
} & ((WithToken extends false ? UserLoginGuest : never) | (WithToken extends true ? UserLoginWithToken : never))
|
} & ((WithToken extends false ? UserLoginGuest : never) | (WithToken extends true ? UserLoginWithToken : never))
|
||||||
|
|
||||||
export interface ElkMasto extends MastoClient {
|
export interface ElkMasto extends MastoClient {
|
||||||
loginTo(user?: UserLogin): Promise<MastoClient>
|
loginTo: LoginTo
|
||||||
loggedIn: Ref<boolean>
|
loggedIn: Ref<boolean>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue