Convert all remaining components into script-setup format
This commit is contained in:
parent
5f2506c170
commit
78dfbfc396
4 changed files with 250 additions and 277 deletions
|
@ -10,7 +10,7 @@
|
||||||
<div class="textarea-group">
|
<div class="textarea-group">
|
||||||
<textarea
|
<textarea
|
||||||
id="content"
|
id="content"
|
||||||
ref="postFormContent"
|
ref="postFormContentRef"
|
||||||
v-model="content"
|
v-model="content"
|
||||||
rows="1"
|
rows="1"
|
||||||
required
|
required
|
||||||
|
@ -29,7 +29,7 @@
|
||||||
<img :src="require('@/assets/feather/paperclip.svg')">
|
<img :src="require('@/assets/feather/paperclip.svg')">
|
||||||
<input
|
<input
|
||||||
type="file"
|
type="file"
|
||||||
ref="attachmentUploadInput"
|
ref="attachmentUploadInputRef"
|
||||||
accept="image/*"
|
accept="image/*"
|
||||||
style="display: none;"
|
style="display: none;"
|
||||||
@change="onAttachmentUpload($event)"
|
@change="onAttachmentUpload($event)"
|
||||||
|
@ -85,9 +85,9 @@
|
||||||
</form>
|
</form>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script setup lang="ts">
|
||||||
import { Options, Vue, setup } from "vue-class-component"
|
import { nextTick, onMounted } from "vue"
|
||||||
import { Prop } from "vue-property-decorator"
|
import { $, $computed, $ref } from "vue/macros"
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Visibility,
|
Visibility,
|
||||||
|
@ -107,127 +107,119 @@ import { setupAutoResize, triggerResize } from "@/utils/autoresize"
|
||||||
import { renderMarkdownLite } from "@/utils/markdown"
|
import { renderMarkdownLite } from "@/utils/markdown"
|
||||||
import { fileToDataUrl, dataUrlToBase64 } from "@/utils/upload"
|
import { fileToDataUrl, dataUrlToBase64 } from "@/utils/upload"
|
||||||
|
|
||||||
@Options({
|
const visibilityMap = Object.entries(VISIBILITY_MAP)
|
||||||
components: {
|
|
||||||
Avatar,
|
const { currentUser, ensureAuthToken } = $(useCurrentUser())
|
||||||
VisibilityIcon,
|
const { instance, getActorAddress } = $(useInstanceInfo())
|
||||||
},
|
|
||||||
|
/* eslint-disable-next-line no-undef */
|
||||||
|
const props = defineProps<{
|
||||||
|
inReplyTo: Post | null,
|
||||||
|
}>()
|
||||||
|
|
||||||
|
/* eslint-disable-next-line no-undef, func-call-spacing */
|
||||||
|
const emit = defineEmits<{
|
||||||
|
(event: "post-created", post: Post): void,
|
||||||
|
}>()
|
||||||
|
|
||||||
|
const postFormContentRef = $ref<HTMLTextAreaElement | null>(null)
|
||||||
|
const attachmentUploadInputRef = $ref<HTMLInputElement | null>(null)
|
||||||
|
|
||||||
|
let content = $ref("")
|
||||||
|
let attachment = $ref<Attachment | null>(null)
|
||||||
|
let visibility = $ref(Visibility.Public)
|
||||||
|
|
||||||
|
let visibilityMenuVisible = $ref(false)
|
||||||
|
let errorMessage = $ref<string | null>(null)
|
||||||
|
|
||||||
|
const author = $computed<User | null>(() => {
|
||||||
|
return currentUser
|
||||||
})
|
})
|
||||||
export default class PostEditor extends Vue {
|
|
||||||
|
|
||||||
@Prop()
|
if (props.inReplyTo) {
|
||||||
inReplyTo: Post | null = null
|
const mentions: Mention[] = [
|
||||||
|
props.inReplyTo.account,
|
||||||
|
...props.inReplyTo.mentions,
|
||||||
|
]
|
||||||
|
content = mentions
|
||||||
|
.filter(mention => mention.id !== currentUser?.id)
|
||||||
|
.map(mention => "@" + getActorAddress(mention))
|
||||||
|
// Remove duplicates
|
||||||
|
.filter((mention, index, mentions) => mentions.indexOf(mention) === index)
|
||||||
|
.join(" ")
|
||||||
|
}
|
||||||
|
if (props.inReplyTo && props.inReplyTo.visibility !== Visibility.Public) {
|
||||||
|
visibility = Visibility.Direct
|
||||||
|
}
|
||||||
|
|
||||||
content = ""
|
onMounted(() => {
|
||||||
attachment: Attachment | null = null
|
if (postFormContentRef) {
|
||||||
visibility = Visibility.Public
|
setupAutoResize(postFormContentRef)
|
||||||
mentions: string[] = []
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
function selectAttachment() {
|
||||||
|
if (attachmentUploadInputRef) {
|
||||||
|
attachmentUploadInputRef.click()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function onAttachmentUpload(event: Event) {
|
||||||
|
const files = (event.target as HTMLInputElement).files
|
||||||
|
if (!files) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const imageDataUrl = await fileToDataUrl(files[0])
|
||||||
|
const imageBase64 = dataUrlToBase64(imageDataUrl)
|
||||||
|
attachment = await uploadAttachment(
|
||||||
|
ensureAuthToken(),
|
||||||
|
imageBase64,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleVisibilityMenu() {
|
||||||
|
visibilityMenuVisible = !visibilityMenuVisible
|
||||||
|
}
|
||||||
|
|
||||||
|
function hideVisibilityMenu() {
|
||||||
visibilityMenuVisible = false
|
visibilityMenuVisible = false
|
||||||
errorMessage: string | null = null
|
}
|
||||||
|
|
||||||
$refs!: {
|
function getCharacterCount(): number {
|
||||||
postFormContent: HTMLTextAreaElement,
|
if (!instance) {
|
||||||
attachmentUploadInput: HTMLInputElement,
|
return 0
|
||||||
}
|
}
|
||||||
|
return (instance.post_character_limit - content.length)
|
||||||
|
}
|
||||||
|
|
||||||
private store = setup(() => {
|
async function publish() {
|
||||||
const { currentUser, ensureAuthToken } = useCurrentUser()
|
const contentRendered = renderMarkdownLite(content)
|
||||||
const { instance, getActorAddress } = useInstanceInfo()
|
const postData = {
|
||||||
return { currentUser, ensureAuthToken, instance, getActorAddress }
|
content: contentRendered,
|
||||||
})
|
in_reply_to_id: props.inReplyTo ? props.inReplyTo.id : null,
|
||||||
|
visibility: visibility,
|
||||||
get author(): User | null {
|
mentions: [],
|
||||||
return this.store.currentUser
|
|
||||||
}
|
}
|
||||||
|
let post
|
||||||
created() {
|
try {
|
||||||
if (this.inReplyTo) {
|
post = await createPost(
|
||||||
const mentions: Mention[] = [
|
ensureAuthToken(),
|
||||||
this.inReplyTo.account,
|
postData,
|
||||||
...this.inReplyTo.mentions,
|
attachment,
|
||||||
]
|
|
||||||
this.content = mentions
|
|
||||||
.filter(mention => mention.id !== this.store.currentUser?.id)
|
|
||||||
.map(mention => "@" + this.store.getActorAddress(mention))
|
|
||||||
// Remove duplicates
|
|
||||||
.filter((mention, index, mentions) => mentions.indexOf(mention) === index)
|
|
||||||
.join(" ")
|
|
||||||
}
|
|
||||||
if (this.inReplyTo && this.inReplyTo.visibility !== Visibility.Public) {
|
|
||||||
this.visibility = Visibility.Direct
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mounted() {
|
|
||||||
setupAutoResize(this.$refs.postFormContent)
|
|
||||||
}
|
|
||||||
|
|
||||||
selectAttachment() {
|
|
||||||
this.$refs.attachmentUploadInput.click()
|
|
||||||
}
|
|
||||||
|
|
||||||
async onAttachmentUpload(event: Event) {
|
|
||||||
const files = (event.target as HTMLInputElement).files
|
|
||||||
if (!files) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const imageDataUrl = await fileToDataUrl(files[0])
|
|
||||||
const imageBase64 = dataUrlToBase64(imageDataUrl)
|
|
||||||
this.attachment = await uploadAttachment(
|
|
||||||
this.store.ensureAuthToken(),
|
|
||||||
imageBase64,
|
|
||||||
)
|
)
|
||||||
|
} catch (error: any) {
|
||||||
|
errorMessage = error.message
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
// Refresh editor
|
||||||
get visibilityMap() {
|
errorMessage = null
|
||||||
return Object.entries(VISIBILITY_MAP)
|
attachment = null
|
||||||
|
content = ""
|
||||||
|
if (postFormContentRef) {
|
||||||
|
await nextTick()
|
||||||
|
triggerResize(postFormContentRef)
|
||||||
}
|
}
|
||||||
|
emit("post-created", post)
|
||||||
toggleVisibilityMenu() {
|
|
||||||
this.visibilityMenuVisible = !this.visibilityMenuVisible
|
|
||||||
}
|
|
||||||
|
|
||||||
hideVisibilityMenu() {
|
|
||||||
this.visibilityMenuVisible = false
|
|
||||||
}
|
|
||||||
|
|
||||||
getCharacterCount(): number {
|
|
||||||
if (!this.store.instance) {
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
return (this.store.instance.post_character_limit - this.content.length)
|
|
||||||
}
|
|
||||||
|
|
||||||
async publish() {
|
|
||||||
const content = renderMarkdownLite(this.content)
|
|
||||||
const postData = {
|
|
||||||
content,
|
|
||||||
in_reply_to_id: this.inReplyTo ? this.inReplyTo.id : null,
|
|
||||||
visibility: this.visibility,
|
|
||||||
mentions: this.mentions,
|
|
||||||
}
|
|
||||||
let post
|
|
||||||
try {
|
|
||||||
post = await createPost(
|
|
||||||
this.store.ensureAuthToken(),
|
|
||||||
postData,
|
|
||||||
this.attachment,
|
|
||||||
)
|
|
||||||
} catch (error: any) {
|
|
||||||
this.errorMessage = error.message
|
|
||||||
return
|
|
||||||
}
|
|
||||||
// Refresh editor
|
|
||||||
this.errorMessage = null
|
|
||||||
this.attachment = null
|
|
||||||
this.content = ""
|
|
||||||
this.$nextTick(() => {
|
|
||||||
triggerResize(this.$refs.postFormContent)
|
|
||||||
})
|
|
||||||
this.$emit("post-created", post)
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
<template>
|
<template>
|
||||||
<sidebar-layout>
|
<sidebar-layout>
|
||||||
<template #content>
|
<template #content>
|
||||||
<post-editor @post-created="insertPost"></post-editor>
|
<post-editor :in-reply-to="null" @post-created="insertPost"></post-editor>
|
||||||
<post-list :posts="posts" @load-next-page="loadNextPage"></post-list>
|
<post-list :posts="posts" @load-next-page="loadNextPage"></post-list>
|
||||||
</template>
|
</template>
|
||||||
</sidebar-layout>
|
</sidebar-layout>
|
||||||
|
|
|
@ -20,7 +20,12 @@
|
||||||
<div class="form-title">Want to join?</div>
|
<div class="form-title">Want to join?</div>
|
||||||
<div class="form-control">
|
<div class="form-control">
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<input id="username" v-model="username" required placeholder="Username">
|
<input
|
||||||
|
id="username"
|
||||||
|
v-model="username"
|
||||||
|
required
|
||||||
|
placeholder="Username"
|
||||||
|
>
|
||||||
<div class="addon">@{{ instance.uri }}</div>
|
<div class="addon">@{{ instance.uri }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-message">Only letters, numbers and underscores are allowed.</div>
|
<div class="form-message">Only letters, numbers and underscores are allowed.</div>
|
||||||
|
@ -48,111 +53,97 @@
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script setup lang="ts">
|
||||||
import { Options, Vue, setup } from "vue-class-component"
|
import { $, $ref } from "vue/macros"
|
||||||
|
import { useRouter } from "vue-router"
|
||||||
|
|
||||||
import {
|
import {
|
||||||
createUser,
|
createUser,
|
||||||
getAccessToken,
|
getAccessToken,
|
||||||
getCurrentUser,
|
getCurrentUser,
|
||||||
} from "@/api/users"
|
} from "@/api/users"
|
||||||
import { InstanceInfo } from "@/api/instance"
|
|
||||||
import Loader from "@/components/Loader.vue"
|
import Loader from "@/components/Loader.vue"
|
||||||
import { useInstanceInfo } from "@/store/instance"
|
import { useInstanceInfo } from "@/store/instance"
|
||||||
import { useCurrentUser } from "@/store/user"
|
import { useCurrentUser } from "@/store/user"
|
||||||
import { createEip4361_SignedMessage, getWallet } from "@/utils/ethereum"
|
import { createEip4361_SignedMessage, getWallet } from "@/utils/ethereum"
|
||||||
|
|
||||||
@Options({
|
const router = useRouter()
|
||||||
components: { Loader },
|
const { setCurrentUser, setAuthToken } = useCurrentUser()
|
||||||
})
|
const { instance } = $(useInstanceInfo())
|
||||||
export default class LandingPage extends Vue {
|
|
||||||
|
|
||||||
username = ""
|
const username = $ref("")
|
||||||
inviteCode: string | null = null
|
const inviteCode = $ref<string | null>(null)
|
||||||
|
let isLoading = $ref(false)
|
||||||
|
let loginErrorMessage = $ref<string | null>(null)
|
||||||
|
let registrationErrorMessage = $ref<string | null>(null)
|
||||||
|
|
||||||
|
async function register() {
|
||||||
|
registrationErrorMessage = null
|
||||||
|
if (!instance) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const instanceHost = instance.uri
|
||||||
|
const loginMessage = instance.login_message
|
||||||
|
const signer = await getWallet()
|
||||||
|
if (!signer) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const { message, signature } = await createEip4361_SignedMessage(
|
||||||
|
signer,
|
||||||
|
instanceHost,
|
||||||
|
loginMessage,
|
||||||
|
)
|
||||||
|
isLoading = true
|
||||||
|
let user
|
||||||
|
let authToken
|
||||||
|
try {
|
||||||
|
user = await createUser({
|
||||||
|
username: username,
|
||||||
|
message,
|
||||||
|
signature,
|
||||||
|
invite_code: inviteCode,
|
||||||
|
})
|
||||||
|
authToken = await getAccessToken(message, signature)
|
||||||
|
} catch (error: any) {
|
||||||
|
isLoading = false
|
||||||
|
registrationErrorMessage = error.message
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setCurrentUser(user)
|
||||||
|
setAuthToken(authToken)
|
||||||
isLoading = false
|
isLoading = false
|
||||||
loginErrorMessage: string | null = null
|
router.push({ name: "home" })
|
||||||
registrationErrorMessage: string | null = null
|
|
||||||
|
|
||||||
private store = setup(() => {
|
|
||||||
const { setCurrentUser, setAuthToken } = useCurrentUser()
|
|
||||||
const { instance } = useInstanceInfo()
|
|
||||||
return { setCurrentUser, setAuthToken, instance }
|
|
||||||
})
|
|
||||||
|
|
||||||
get instance(): InstanceInfo | null {
|
|
||||||
return this.store.instance
|
|
||||||
}
|
|
||||||
|
|
||||||
async register() {
|
|
||||||
this.registrationErrorMessage = null
|
|
||||||
if (!this.store.instance) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const instanceHost = this.store.instance.uri
|
|
||||||
const loginMessage = this.store.instance.login_message
|
|
||||||
const signer = await getWallet()
|
|
||||||
if (!signer) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const { message, signature } = await createEip4361_SignedMessage(
|
|
||||||
signer,
|
|
||||||
instanceHost,
|
|
||||||
loginMessage,
|
|
||||||
)
|
|
||||||
this.isLoading = true
|
|
||||||
let user
|
|
||||||
let authToken
|
|
||||||
try {
|
|
||||||
user = await createUser({
|
|
||||||
username: this.username,
|
|
||||||
message,
|
|
||||||
signature,
|
|
||||||
invite_code: this.inviteCode,
|
|
||||||
})
|
|
||||||
authToken = await getAccessToken(message, signature)
|
|
||||||
} catch (error: any) {
|
|
||||||
this.isLoading = false
|
|
||||||
this.registrationErrorMessage = error.message
|
|
||||||
return
|
|
||||||
}
|
|
||||||
this.store.setCurrentUser(user)
|
|
||||||
this.store.setAuthToken(authToken)
|
|
||||||
this.isLoading = false
|
|
||||||
this.$router.push({ name: "home" })
|
|
||||||
}
|
|
||||||
|
|
||||||
async login() {
|
|
||||||
this.loginErrorMessage = null
|
|
||||||
if (!this.store.instance) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const instanceHost = this.store.instance.uri
|
|
||||||
const loginMessage = this.store.instance.login_message
|
|
||||||
const signer = await getWallet()
|
|
||||||
if (!signer) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const { message, signature } = await createEip4361_SignedMessage(
|
|
||||||
signer,
|
|
||||||
instanceHost,
|
|
||||||
loginMessage,
|
|
||||||
)
|
|
||||||
let user
|
|
||||||
let authToken
|
|
||||||
try {
|
|
||||||
authToken = await getAccessToken(message, signature)
|
|
||||||
user = await getCurrentUser(authToken)
|
|
||||||
} catch (error: any) {
|
|
||||||
this.loginErrorMessage = error.message
|
|
||||||
return
|
|
||||||
}
|
|
||||||
this.store.setCurrentUser(user)
|
|
||||||
this.store.setAuthToken(authToken)
|
|
||||||
this.$router.push({ name: "home" })
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function login() {
|
||||||
|
loginErrorMessage = null
|
||||||
|
if (!instance) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const instanceHost = instance.uri
|
||||||
|
const loginMessage = instance.login_message
|
||||||
|
const signer = await getWallet()
|
||||||
|
if (!signer) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const { message, signature } = await createEip4361_SignedMessage(
|
||||||
|
signer,
|
||||||
|
instanceHost,
|
||||||
|
loginMessage,
|
||||||
|
)
|
||||||
|
let user
|
||||||
|
let authToken
|
||||||
|
try {
|
||||||
|
authToken = await getAccessToken(message, signature)
|
||||||
|
user = await getCurrentUser(authToken)
|
||||||
|
} catch (error: any) {
|
||||||
|
loginErrorMessage = error.message
|
||||||
|
return
|
||||||
|
}
|
||||||
|
setCurrentUser(user)
|
||||||
|
setAuthToken(authToken)
|
||||||
|
router.push({ name: "home" })
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
|
|
|
@ -50,8 +50,10 @@
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script setup lang="ts">
|
||||||
import { Options, Vue, setup } from "vue-class-component"
|
import { onMounted } from "vue"
|
||||||
|
import { $, $ref, $computed } from "vue/macros"
|
||||||
|
import { useRoute, useRouter } from "vue-router"
|
||||||
|
|
||||||
import { DateTime } from "luxon"
|
import { DateTime } from "luxon"
|
||||||
|
|
||||||
|
@ -61,79 +63,67 @@ import Avatar from "@/components/Avatar.vue"
|
||||||
import { useInstanceInfo } from "@/store/instance"
|
import { useInstanceInfo } from "@/store/instance"
|
||||||
import { useCurrentUser } from "@/store/user"
|
import { useCurrentUser } from "@/store/user"
|
||||||
|
|
||||||
@Options({
|
const route = useRoute()
|
||||||
components: {
|
const router = useRouter()
|
||||||
Avatar,
|
const { currentUser, authToken } = $(useCurrentUser())
|
||||||
},
|
const { instance, getActorAddress } = $(useInstanceInfo())
|
||||||
|
|
||||||
|
let post = $ref<Post | null>(null)
|
||||||
|
let token = $ref<TokenMetadata | null>(null)
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
post = await getPost(
|
||||||
|
authToken,
|
||||||
|
route.params.postId as string,
|
||||||
|
)
|
||||||
|
if (metadataUrl) {
|
||||||
|
token = await getTokenMetadata(metadataUrl)
|
||||||
|
}
|
||||||
})
|
})
|
||||||
export default class PostOverlay extends Vue {
|
|
||||||
|
|
||||||
post: Post | null = null
|
|
||||||
token: TokenMetadata | null = null
|
|
||||||
|
|
||||||
private store = setup(() => {
|
|
||||||
const { currentUser, authToken } = useCurrentUser()
|
|
||||||
const { instance, getActorAddress } = useInstanceInfo()
|
|
||||||
return { currentUser, authToken, instance, getActorAddress }
|
|
||||||
})
|
|
||||||
|
|
||||||
async created() {
|
|
||||||
this.post = await getPost(
|
|
||||||
this.store.authToken,
|
|
||||||
this.$route.params.postId as string,
|
|
||||||
)
|
|
||||||
const metadataUrl = this.metadataUrl
|
|
||||||
if (metadataUrl) {
|
|
||||||
this.token = await getTokenMetadata(metadataUrl)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
canGoBack(): boolean {
|
|
||||||
return this.store.currentUser !== null
|
|
||||||
}
|
|
||||||
|
|
||||||
goBack() {
|
|
||||||
this.$router.back()
|
|
||||||
}
|
|
||||||
|
|
||||||
get actorAddress(): string {
|
|
||||||
if (!this.post) {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return this.store.getActorAddress(this.post.account)
|
|
||||||
}
|
|
||||||
|
|
||||||
get transactionUrl(): string | null {
|
|
||||||
const explorerUrl = this.store.instance?.blockchain_explorer_url
|
|
||||||
if (!explorerUrl || !this.post?.token_tx_id) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
return `${explorerUrl}/tx/0x${this.post.token_tx_id}`
|
|
||||||
}
|
|
||||||
|
|
||||||
get metadataUrl(): string | null {
|
|
||||||
const gatewayUrl = this.store.instance?.ipfs_gateway_url
|
|
||||||
if (!gatewayUrl || !this.post) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
return `${gatewayUrl}/ipfs/${this.post.ipfs_cid}`
|
|
||||||
}
|
|
||||||
|
|
||||||
get imageUrl(): string | null {
|
|
||||||
const gatewayUrl = this.store.instance?.ipfs_gateway_url
|
|
||||||
if (!gatewayUrl || !this.token) {
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
return this.token.image.replace("ipfs://", `${gatewayUrl}/ipfs/`)
|
|
||||||
}
|
|
||||||
|
|
||||||
formatDate(isoDate: string): string {
|
|
||||||
const date = DateTime.fromISO(isoDate)
|
|
||||||
return date.toLocaleString(DateTime.DATE_FULL)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
function canGoBack(): boolean {
|
||||||
|
return currentUser !== null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function goBack() {
|
||||||
|
router.back()
|
||||||
|
}
|
||||||
|
|
||||||
|
const actorAddress = $computed<string>(() => {
|
||||||
|
if (!post) {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return getActorAddress(post.account)
|
||||||
|
})
|
||||||
|
|
||||||
|
const transactionUrl = $computed<string | null>(() => {
|
||||||
|
const explorerUrl = instance?.blockchain_explorer_url
|
||||||
|
if (!explorerUrl || !post?.token_tx_id) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
return `${explorerUrl}/tx/0x${post.token_tx_id}`
|
||||||
|
})
|
||||||
|
|
||||||
|
const metadataUrl = $computed<string | null>(() => {
|
||||||
|
const gatewayUrl = instance?.ipfs_gateway_url
|
||||||
|
if (!gatewayUrl || !post) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
return `${gatewayUrl}/ipfs/${post.ipfs_cid}`
|
||||||
|
})
|
||||||
|
|
||||||
|
const imageUrl = $computed<string | null>(() => {
|
||||||
|
const gatewayUrl = instance?.ipfs_gateway_url
|
||||||
|
if (!gatewayUrl || !token) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
return token.image.replace("ipfs://", `${gatewayUrl}/ipfs/`)
|
||||||
|
})
|
||||||
|
|
||||||
|
function formatDate(isoDate: string): string {
|
||||||
|
const date = DateTime.fromISO(isoDate)
|
||||||
|
return date.toLocaleString(DateTime.DATE_FULL)
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped lang="scss">
|
<style scoped lang="scss">
|
||||||
|
|
Loading…
Reference in a new issue