Merge branch 'new-CounterFire' of http://git.kingsome.cn/huangjinming/CounterFireGames into new-CounterFire
This commit is contained in:
commit
b29c4e55ba
@ -132,6 +132,20 @@ export class BlockChain {
|
|||||||
|
|
||||||
return token+suffix
|
return token+suffix
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* 如果是用passport登录的, 直接返回store中的token, 否则调用passport的getAccessToken
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
async passportToken() {
|
||||||
|
if (this.store.walletType == 3) {
|
||||||
|
return this.store.token
|
||||||
|
}
|
||||||
|
if (!this.passportProvider) {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
const res = await new PassportWallet().getAccessToken();
|
||||||
|
return res.token
|
||||||
|
}
|
||||||
|
|
||||||
async logout() {
|
async logout() {
|
||||||
if (this.store.walletType != 3 && this.passportProvider) {
|
if (this.store.walletType != 3 && this.passportProvider) {
|
||||||
@ -148,11 +162,12 @@ export class BlockChain {
|
|||||||
/**
|
/**
|
||||||
* 检查并切换到目标链, 各上链前须调用该方法
|
* 检查并切换到目标链, 各上链前须调用该方法
|
||||||
*/
|
*/
|
||||||
async checkAndChangeChain() {
|
async checkAndChangeChain(targetChainId) {
|
||||||
|
targetChainId = targetChainId || cfgChainId;
|
||||||
let chainId = await this.getChainId();
|
let chainId = await this.getChainId();
|
||||||
if (chainId !== cfgChainId) {
|
if (chainId !== targetChainId) {
|
||||||
console.log(`current chain: ${chainId}, want: ${cfgChainId}`)
|
console.log(`current chain: ${chainId}, want: ${targetChainId}`)
|
||||||
chainId = await switchEthereumChain(this.web3Provider.provider, cfgChainId);
|
chainId = await switchEthereumChain(this.web3Provider.provider, targetChainId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -175,4 +190,26 @@ export class BlockChain {
|
|||||||
await this.appendPassport();
|
await this.appendPassport();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async selectAddress({title, subTitle, targetChainId}) {
|
||||||
|
const initData = {}
|
||||||
|
initData[this.store.walletType] = this.store.address
|
||||||
|
const rewardModal = createModal(WalletSelectModel, {
|
||||||
|
title: title || 'Select Address',
|
||||||
|
message: subTitle || 'Please select the address you want to use',
|
||||||
|
initData: { 1: 'MetaMask', 2: 'OKX Wallet', 3: 'Passport' }
|
||||||
|
});
|
||||||
|
const { errcode, errmsg, walletInstance, provider, accounts } = await rewardModal.show();
|
||||||
|
if (errcode) {
|
||||||
|
console.log(`select address result : ${errmsg}`);
|
||||||
|
throw new Error(errmsg);
|
||||||
|
}
|
||||||
|
targetChainId = targetChainId || cfgChainId;
|
||||||
|
let chainId = await walletInstance.getChainId();
|
||||||
|
if (chainId !== targetChainId) {
|
||||||
|
console.log(`current chain: ${chainId}, want: ${targetChainId}`)
|
||||||
|
chainId = await switchEthereumChain(provider.provider, targetChainId);
|
||||||
|
}
|
||||||
|
return { provider, address: accounts[0] };
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
<div class="modal-bg" @click="cancelSelect"></div>
|
<div class="modal-bg" @click="cancelSelect"></div>
|
||||||
<div class="modal-content" :class="{ mobile: 'mobile' }">
|
<div class="modal-content" :class="{ mobile: 'mobile' }">
|
||||||
<div class="modal-title">
|
<div class="modal-title">
|
||||||
<div>Please connect your wallet</div>
|
<div>{{title||'Please connect your wallet'}}</div>
|
||||||
<img
|
<img
|
||||||
src=""
|
src=""
|
||||||
alt="close"
|
alt="close"
|
||||||
@ -11,7 +11,7 @@
|
|||||||
@click="cancelSelect"/>
|
@click="cancelSelect"/>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-sub-title">
|
<div class="modal-sub-title">
|
||||||
Connect your wallet to start your Counter Fire journey
|
{{subTitle || 'Connect your wallet to start your Counter Fire journey'}}
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<div
|
<div
|
||||||
@ -39,19 +39,29 @@ import { allProviders } from "@/components/chain/BlockChain"
|
|||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
visible: ref(Boolean),
|
visible: ref(Boolean),
|
||||||
close: Function,
|
close: Function,
|
||||||
|
title: ref(String),
|
||||||
|
subTitle: ref(String),
|
||||||
|
initData: Object
|
||||||
});
|
});
|
||||||
|
|
||||||
const currentDatas = computed(() => {
|
const currentDatas = computed(() => {
|
||||||
let list = ALL_PROVIDERS.map((item) => {
|
let list = ALL_PROVIDERS.map((item) => {
|
||||||
const Provider = allProviders[item.id];
|
const Provider = allProviders[item.id];
|
||||||
const installed = new Provider().installed;
|
const installed = new Provider().installed;
|
||||||
|
const initData = props.initData;
|
||||||
|
let current = null;
|
||||||
|
if (initData) {
|
||||||
|
current = initData[item.id];
|
||||||
|
}
|
||||||
|
|
||||||
|
const tip = installed ? current?.tip || item.tip || '' : "Click to Install";
|
||||||
return {
|
return {
|
||||||
id: item.id,
|
id: item.id,
|
||||||
name: item.name,
|
name: item.name,
|
||||||
logo: item.logo,
|
logo: item.logo,
|
||||||
installed,
|
installed,
|
||||||
downloadUrl: item.downloadUrl,
|
downloadUrl: item.downloadUrl,
|
||||||
tip: installed ? item.tip || "" : "Click to Install",
|
tip,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -60,9 +70,10 @@ const currentDatas = computed(() => {
|
|||||||
|
|
||||||
async function selectWallet(id) {
|
async function selectWallet(id) {
|
||||||
const Provider = allProviders[id];
|
const Provider = allProviders[id];
|
||||||
const { provider, accounts } = await new Provider().web3Provider();
|
const walletInstance = new Provider()
|
||||||
|
const { provider, accounts } = await walletInstance.web3Provider();
|
||||||
console.log(accounts)
|
console.log(accounts)
|
||||||
hideModal({errcode: 0, provider, wallet: id, accounts});
|
hideModal({errcode: 0, provider, wallet: id, walletInstance, accounts});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -76,8 +87,9 @@ async function cardClicked(id) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
window.open(data.downloadUrl, '_blank');
|
window.open(data.downloadUrl, '_blank');
|
||||||
}
|
} else {
|
||||||
await selectWallet(id);
|
await selectWallet(id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function cancelSelect() {
|
function cancelSelect() {
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
|
|
||||||
import { ethers } from 'ethers'
|
import { ethers } from 'ethers'
|
||||||
|
import { apiUnlockOrMint } from '@/utils/marketplace'
|
||||||
|
|
||||||
const lockAbi = [
|
const lockAbi = [
|
||||||
'function lock(address nft, address to, uint256[] tokenIds) external'
|
'function lock(address nft, address to, uint256[] tokenIds) external',
|
||||||
|
'function unlockOrMint(address nft, tuple[] nftList, uint256 signTime, uint256 saltNonce, bytes signature) external'
|
||||||
]
|
]
|
||||||
|
|
||||||
const erc721Abi = [
|
const erc721Abi = [
|
||||||
@ -40,4 +42,56 @@ export class Locker {
|
|||||||
await this.bc.web3Provider.waitForTransaction(res.hash)
|
await this.bc.web3Provider.waitForTransaction(res.hash)
|
||||||
return res.hash
|
return res.hash
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
async sendUnlockOrMint(provider, {to, data}) {
|
||||||
|
|
||||||
|
const txHash = await web3Provider.request({
|
||||||
|
method: 'eth_sendTransaction',
|
||||||
|
params: [{
|
||||||
|
to,
|
||||||
|
data
|
||||||
|
}]
|
||||||
|
})
|
||||||
|
console.log(txHash)
|
||||||
|
return txHash
|
||||||
|
}
|
||||||
|
|
||||||
|
// 游戏内资产上链, 用于解锁或铸造
|
||||||
|
// 创世英雄, 普通英雄, 金砖
|
||||||
|
// imtbl上unlock必须使用passport的provider
|
||||||
|
async unlockOrMintGameNft(nft, tokenIds) {
|
||||||
|
console.log('unlock nft', nft, tokenIds)
|
||||||
|
await this.bc.checkPassportLogin();
|
||||||
|
await this.bc.checkAndChangeChain();
|
||||||
|
const preDatas = {
|
||||||
|
net_id: import.meta.env.VUE_APP_NET_ID,
|
||||||
|
contract_address: nft,
|
||||||
|
tokens: tokenIds.map(tokenId => {return { tokenId }}),
|
||||||
|
}
|
||||||
|
const passportToken = await this.bc.passportToken()
|
||||||
|
const { errcode, errmsg, trans_req } = await apiUnlockOrMint(preDatas, passportToken)
|
||||||
|
if (errcode) {
|
||||||
|
throw new Error(errmsg)
|
||||||
|
}
|
||||||
|
const web3Provider = this.bc.passportProvider || this.bc.web3Provider
|
||||||
|
return this.sendUnlockOrMint(web3Provider, trans_req)
|
||||||
|
}
|
||||||
|
// 游戏内资产上链, 只用于mint
|
||||||
|
// 该方法会显示一个确认弹窗, 由用户选择mint到哪个地址
|
||||||
|
async mintNft(nft, tokenIds) {
|
||||||
|
console.log('mint nft', nft, tokenIds)
|
||||||
|
const { provider, address } = await this.bc.selectAddress()
|
||||||
|
const preDatas = {
|
||||||
|
net_id: import.meta.env.VUE_APP_NET_ID,
|
||||||
|
to: address,
|
||||||
|
contract_address: nft,
|
||||||
|
tokens: tokenIds.map(tokenId => {return { tokenId }}),
|
||||||
|
}
|
||||||
|
const passportToken = await this.bc.passportToken()
|
||||||
|
const { errcode, errmsg, trans_req } = await apiUnlockOrMint(preDatas, passportToken)
|
||||||
|
if (errcode) {
|
||||||
|
throw new Error(errmsg)
|
||||||
|
}
|
||||||
|
return this.sendUnlockOrMint(provider, trans_req)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -298,5 +298,13 @@ export const AllChains = [
|
|||||||
id: 13473,
|
id: 13473,
|
||||||
symbol: 'tIMX',
|
symbol: 'tIMX',
|
||||||
explorerurl: 'https://explorer.testnet.immutable.com'
|
explorerurl: 'https://explorer.testnet.immutable.com'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Sepolia Testnet',
|
||||||
|
type: 'Testnet',
|
||||||
|
rpc: 'https://rpc.sepolia.org',
|
||||||
|
id: 11155111,
|
||||||
|
symbol: 'ETH',
|
||||||
|
explorerurl: 'https://sepolia.etherscan.io'
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -8,8 +8,8 @@ const net_id = import.meta.env.VUE_APP_NET_ID
|
|||||||
|
|
||||||
const toJson = res => res.json();
|
const toJson = res => res.json();
|
||||||
|
|
||||||
const httpPost = async (url, data) => {
|
const httpPost = async (url, data, token) => {
|
||||||
const token = await new BlockChain().token();
|
token = token || await new BlockChain().token();
|
||||||
let headers = {"Content-Type": "application/json"};
|
let headers = {"Content-Type": "application/json"};
|
||||||
// let token = token;
|
// let token = token;
|
||||||
if (token) {
|
if (token) {
|
||||||
@ -23,8 +23,8 @@ const httpPost = async (url, data) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const httpGet = async (url, data) => {
|
const httpGet = async (url, data, token) => {
|
||||||
const token = await new BlockChain().token();
|
token = token || await new BlockChain().token();
|
||||||
let headers = {"Content-Type": "application/json"};
|
let headers = {"Content-Type": "application/json"};
|
||||||
if (token) {
|
if (token) {
|
||||||
headers['Authorization'] = `Bearer ${token}`;
|
headers['Authorization'] = `Bearer ${token}`;
|
||||||
@ -124,3 +124,8 @@ export const apiHeroList = async(data) => {
|
|||||||
const url = `${API_BASE}/api/ingame/asset/hero/list`
|
const url = `${API_BASE}/api/ingame/asset/hero/list`
|
||||||
return httpPost(url, data)
|
return httpPost(url, data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const apiUnlockOrMint = async (data, token) => {
|
||||||
|
const url = `${API_BASE}/api/nft/stacking/unlock`
|
||||||
|
return httpPost(url, data, token)
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user