-
Mateusz Tyszczak authoredMateusz Tyszczak authored
MetamaskConnect.vue 15.80 KiB
<script setup lang="ts">
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from "@/components/ui/card";
import { Button } from "@/components/ui/button";
import { Input } from '@/components/ui/input'
import { Label } from '@/components/ui/label'
import { ref, onMounted } from 'vue';
import step1 from "@/assets/icons/wallets/metamask/step1.webp";
import step2 from "@/assets/icons/wallets/metamask/step2.webp";
import { mdiClose } from '@mdi/js';
import { UsedWallet, getWalletIcon } from '@/stores/settings.store';
import { useMetamaskStore } from "@/stores/metamask.store";
import { Combobox, ComboboxAnchor, ComboboxTrigger, ComboboxEmpty, ComboboxGroup, ComboboxInput, ComboboxItem, ComboboxItemIndicator, ComboboxList } from '@/components/ui/combobox';
import { Check, Search } from 'lucide-vue-next';
import { Separator } from '@/components/ui/separator';
import PublicKey from '@/components/hive/PublicKey.vue';
import { Checkbox } from '@/components/ui/checkbox'
import { getWax } from '@/stores/wax.store';
import type { TRole } from "@hiveio/wax/vite";
const emit = defineEmits(["setaccount", "close"]);
const showUpdateAccountModal = ref(false);
const showCreateAccountModal = ref(false);
const updateAuthType: Record<TRole, boolean> = {
owner: true,
active: true,
posting: true,
memo: true
};
const close = () => {
showUpdateAccountModal.value = false;
showCreateAccountModal.value = false;
emit("close");
};
const metamaskStore = useMetamaskStore();
const isLoading = ref(false);
const errorMsg = ref<string | null>(null);
const accountName = ref<string | null>(null);
const createAccountNameOperation = ref<string | null>(null);
const updateAccountNameOperation = ref<string | null>(null);
const accountsMatchingKeys = ref<string[] | null>(null);
const metamaskPublicKeys = ref<Array<{ role: string; publicKey: string }> | null>(null);
const isMetamaskConnected = ref<boolean>(false);
const isMetamaskSnapInstalled = ref<boolean>(false);
const applyPublicKeys = async () => {
isLoading.value = true;
errorMsg.value = null;
try {
const { publicKeys } = await metamaskStore.call("hive_getPublicKeys", {
keys: [{
role: "owner"
},{
role: "active"
},{
role: "posting"
},{
role: "memo"
}]
}) as any;
metamaskPublicKeys.value = publicKeys;
const wax = await getWax();
const response = await wax.api.account_by_key_api.get_key_references({
keys: publicKeys.map((node: { publicKey: string }) => node.publicKey)
});
accountsMatchingKeys.value = [...new Set(response.accounts.flatMap((node: string[]) => node))] as string[];
} catch (error) {
if (typeof error === "object" && error && "message" in error)
errorMsg.value = error.message as string;
else
errorMsg.value = String(error);
} finally {
isLoading.value = false;
}
}
const connect = async (showError = true) => {
isLoading.value = true;
errorMsg.value = null;
try {
await metamaskStore.connect();
isMetamaskConnected.value = metamaskStore.isConnected;
isMetamaskSnapInstalled.value = metamaskStore.isInstalled!;
if (isMetamaskSnapInstalled.value)
void applyPublicKeys();
} catch (error) {
if (!showError)
return;
if (typeof error === "object" && error && "message" in error)
errorMsg.value = error.message as string;
else
errorMsg.value = String(error);
} finally {
isLoading.value = false;
};
};
const install = async () => {
isLoading.value = true;
errorMsg.value = null;
try {
await metamaskStore.install();
isMetamaskSnapInstalled.value = metamaskStore.isInstalled!;
if (isMetamaskSnapInstalled.value)
void applyPublicKeys();
} catch (error) {
if (typeof error === "object" && error && "message" in error)
errorMsg.value = error.message as string;
else
errorMsg.value = String(error);
} finally {
isLoading.value = false;
};
}
onMounted(() => {
void connect(false);
});
const copyContent = (content: string) => {
navigator.clipboard.writeText(String(content));
};
const generateAccountUpdateTransaction = async(): Promise<string> => {
const wax = await getWax();
const tx = await wax.createTransaction();
const accountName = updateAccountNameOperation.value!.startsWith('@') ? updateAccountNameOperation.value!.slice(1) : updateAccountNameOperation.value!;
const { AccountAuthorityUpdateOperation } = await import("@hiveio/wax/vite");
const op = await AccountAuthorityUpdateOperation.createFor(wax, accountName);
for(const key in updateAuthType) {
if (updateAuthType[key as TRole])
if (key === "memo")
op.role("memo").set(metamaskPublicKeys.value!.find(node => node.role === key)!.publicKey);
else
op.role(key as Exclude<TRole, "memo">).add(metamaskPublicKeys.value!.find(node => node.role === key)!.publicKey);
}
tx.pushOperation(op);
return tx.toApi();
};
const getAccountCreateSigningLink = (): string => {
const accountName = createAccountNameOperation.value!.startsWith('@') ? createAccountNameOperation.value!.slice(1) : createAccountNameOperation.value!;
return `${window.location.protocol}//${window.location.host}/account/create?acc=${accountName}&posting=${
metamaskPublicKeys.value!.find(node => node.role === "posting")!.publicKey
}&active=${
metamaskPublicKeys.value!.find(node => node.role === "active")!.publicKey
}&owner=${
metamaskPublicKeys.value!.find(node => node.role === "owner")!.publicKey
}&memo=${
metamaskPublicKeys.value!.find(node => node.role === "memo")!.publicKey
}`;
};
const getAuthorityUpdateSigningLink = (): string => {
const accountName = updateAccountNameOperation.value!.startsWith('@') ? updateAccountNameOperation.value!.slice(1) : updateAccountNameOperation.value!;
const url = new URL(`${window.location.protocol}//${window.location.host}/account/update?acc=${accountName}`);
for(const key in updateAuthType)
if (updateAuthType[key as TRole])
url.searchParams.set(key, metamaskPublicKeys.value!.find(node => node.role === key)!.publicKey);
return url.toString();
};
const updateAccountName = (value: string | any) => {
if (!value) return;
accountName.value = value.startsWith("@") ? value.slice(1) : value;
};
</script>
<template>
<Card class="w-[350px]">
<CardHeader>
<CardTitle>
<div class="inline-flex justify-between w-full">
<div class="inline-flex items-center">
<img :src="getWalletIcon(UsedWallet.METAMASK)" class="w-[20px] mr-2" />
<span class="mt-[2px]">Metamask Connector</span>
</div>
<Button variant="ghost" size="sm" class="px-2" @click="close">
<svg width="24" height="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path style="fill: hsl(var(--foreground))" :d="mdiClose"/></svg>
</Button>
</div>
</CardTitle>
<CardDescription>Follow these instructions to connect to Metamask</CardDescription>
</CardHeader>
<CardContent class="text-sm">
<div v-if="!isMetamaskConnected" class="space-y-4">
<p>Step 1: Connect your metamask wallet:</p>
<div class="flex justify-center">
<Button :disabled="isLoading" variant="outline" size="lg" class="px-8 py-4 border-[#FF5C16] border-[2px]" @click="connect">
<span class="text-md font-bold">Connect</span>
</Button>
</div>
</div>
<div v-if="!isMetamaskSnapInstalled && !isLoading" class="space-y-4">
<p>Step 2: Install our Hive Wallet snap:</p>
<div class="flex justify-center">
<Button :disabled="isLoading" variant="outline" size="lg" class="px-8 py-4 border-[#FF5C16] border-[2px]" @click="install">
<span class="text-md font-bold">Install snap</span>
</Button>
</div>
</div>
<div v-if="!isMetamaskSnapInstalled && isLoading" class="space-y-4">
<p>Step 3: Connect to our snap server:</p>
<div class="flex justify-center">
<img :src="step1" class="w-[234px] h-[172px] mt-4 border rounded-md p-2 ml-auto mr-auto" />
</div>
<p>Step 4: Confirm snap installation:</p>
<div class="flex justify-center">
<img :src="step2" class="w-[237px] h-[253px] mt-4 border rounded-md p-2 ml-auto mr-auto" />
</div>
</div>
<div v-if="isMetamaskSnapInstalled">
<div v-if="accountsMatchingKeys">
<div v-if="showUpdateAccountModal">
<p class="mb-4">Step 6: Fill in this form in order to create account update operation, replacing memo public key and adding posting, active and owner keys to your account:</p>
<div class="grid mb-2 w-full max-w-sm items-center gap-1.5">
<Label for="metamask_updateAuth_account">Account name</Label>
<Input v-model="updateAccountNameOperation as string" id="metamask_updateAuth_account" />
</div>
<div v-for="key in metamaskPublicKeys" :key="key.publicKey">
<div class="flex items-center p-1">
<Checkbox :id="`metamask_updateAuth_key-${key.role}`" :defaultValue="true" @update:modelValue="value => { updateAuthType[key.role as TRole] = value as boolean }" />
<label :for="`metamask_updateAuth_key-${key.role}`" class="pl-2 w-full flex items-center">
<span class="font-bold">{{ key.role[0].toUpperCase() }}{{ key.role.slice(1) }}</span>
<div class="mx-2 border flex-grow border-[hsl(var(--foreground))] opacity-[0.1]" />
<PublicKey :value="key.publicKey"/>
</label>
</div>
</div>
<div class="flex items-center flex-col">
<Button :disabled="isLoading" @click="copyContent(getAuthorityUpdateSigningLink())" variant="outline" size="lg" class="mt-4 px-8 py-4 border-[#FF5C16] border-[1px]">
<span class="text-md font-bold">Copy signing link</span>
</Button>
<Separator label="Or" class="mt-8" />
<div class="flex justify-center mt-4">
<Button :disabled="isLoading" @click="generateAccountUpdateTransaction()" variant="outline" size="lg" class="px-8 opacity-[0.9] py-4 border-[#FF5C16] border-[1px]">
<span class="text-md font-bold">Copy entire transaction</span>
</Button>
</div>
</div>
</div>
<div v-else-if="showCreateAccountModal">
<p class="mb-4">Step 6: Fill in this form in order to create account create operation with requested metadata. Copy the signing link and send it to someone who already has an account:</p>
<div class="grid mb-2 w-full max-w-sm items-center gap-1.5">
<Label for="metamask_createAuth_account">New account name</Label>
<Input v-model="createAccountNameOperation as string" id="metamask_createAuth_account" />
</div>
<div v-for="key in metamaskPublicKeys" :key="key.publicKey">
<div class="flex items-center p-1">
<span class="font-bold">{{ key.role[0].toUpperCase() }}{{ key.role.slice(1) }}</span>
<div class="mx-2 border flex-grow border-[hsl(var(--foreground))] opacity-[0.1]" />
<PublicKey :value="key.publicKey"/>
</div>
</div>
<div class="flex items-center flex-col">
<Button :disabled="isLoading" @click="copyContent(getAccountCreateSigningLink())" variant="outline" size="lg" class="mt-4 px-8 py-4 border-[#FF5C16] border-[1px]">
<span class="text-md font-bold">Copy signing link</span>
</Button>
</div>
</div>
<div v-else-if="accountsMatchingKeys.length === 0">
<p class="mb-2">Step 5: Import <b>at least one</b> Metamask derived key into your Hive account and re-check for Hive Accounts matching those keys:</p>
<div v-for="key in metamaskPublicKeys" :key="key.publicKey">
<div class="flex items-center p-1">
<span class="font-bold">{{ key.role[0].toUpperCase() }}{{ key.role.slice(1) }}</span>
<div class="mx-2 border flex-grow border-[hsl(var(--foreground))] opacity-[0.1]" />
<PublicKey :value="key.publicKey"/>
</div>
</div>
<div class="flex justify-center mt-3">
<Button :disabled="isLoading" variant="outline" size="lg" class="px-8 py-4 border-[#FF5C16] border-[2px]" @click="applyPublicKeys">
<span class="text-md font-bold">Re-check for Hive Accounts</span>
</Button>
</div>
<Separator label="Or" class="mt-8" />
<div class="flex justify-center mt-4">
<Button :disabled="isLoading" @click="showUpdateAccountModal = true" variant="outline" size="lg" class="px-8 opacity-[0.9] py-4 border-[#FF5C16] border-[1px]">
<span class="text-md font-bold">Update account authority</span>
</Button>
</div>
<div class="flex justify-center mt-4">
<Button :disabled="isLoading" @click="showCreateAccountModal = true" variant="outline" size="lg" class="px-8 opacity-[0.9] py-4 border-[#FF5C16] border-[1px]">
<span class="text-md font-bold">Request account creation</span>
</Button>
</div>
</div>
<div v-else>
<p>Step 6: Connect your Hive account:</p>
<div class="mt-2 flex justify-center space-y-2">
<Combobox by="label" @update:model-value="updateAccountName">
<ComboboxAnchor>
<ComboboxTrigger as-child>
<div class="relative w-full max-w-sm items-center">
<ComboboxInput class="pl-9" :display-value="(val) => val ? `@${val}` : ''" placeholder="Select account" />
<span class="absolute start-0 inset-y-0 flex items-center justify-center px-3">
<Search class="size-4 text-muted-foreground" />
</span>
</div>
</ComboboxTrigger>
</ComboboxAnchor>
<ComboboxList>
<ComboboxEmpty>
No account matching keys found.
</ComboboxEmpty>
<ComboboxGroup>
<ComboboxItem
v-for="account in accountsMatchingKeys"
:key="account"
:value="account"
>
@{{ account }}
<ComboboxItemIndicator>
<Check class="ml-auto h-4 w-4" />
</ComboboxItemIndicator>
</ComboboxItem>
</ComboboxGroup>
</ComboboxList>
</Combobox>
</div>
<div class="flex justify-center mt-3" v-if="accountName">
<Button :disabled="isLoading" variant="outline" size="lg" class="px-8 py-4 border-[#FF5C16] border-[2px]" @click="emit('setaccount', accountName)">
<span class="text-md font-bold">Import</span>
</Button>
</div>
<Separator label="Or" class="mt-8" />
<div class="flex justify-center mt-4">
<Button :disabled="isLoading" variant="outline" size="lg" class="px-8 opacity-[0.9] py-4 border-[#FF5C16] border-[1px]" @click="accountsMatchingKeys = []">
<span class="text-md font-bold">Import different account</span>
</Button>
</div>
</div>
</div>
<div v-else>
<p>Step 5: Loading on-chain accounts...</p>
<p>Please wait</p>
</div>
</div>
</CardContent>
<CardFooter>
<span class="text-red-400" v-if="errorMsg"><span class="font-bold">Error: </span>{{ errorMsg }}</span>
</CardFooter>
</Card>
</template>