Skip to content

Commit

Permalink
add success page to ai minter (#138)
Browse files Browse the repository at this point in the history
* add success page to ai minter

* add success page to ai minter

* add success page to ai minter

* add success page to ai minter

* add success page to ai minter

* add txn hash

* ai minter

* ai minter

* ai minter

* ai minter
  • Loading branch information
rubenmarcus authored Feb 27, 2024
1 parent 94dce59 commit a6f2f12
Show file tree
Hide file tree
Showing 18 changed files with 308 additions and 67 deletions.
6 changes: 3 additions & 3 deletions ai-minter/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,9 @@
},
"dependencies": {
"@hookform/resolvers": "^3.3.2",
"@mintbase-js/react": "0.5.4-beta.0",
"@mintbase-js/sdk": "0.5.4-beta.0",
"@mintbase-js/storage": "0.5.4-beta.0",
"@mintbase-js/react": "0.5.5-beta.2",
"@mintbase-js/sdk": "0.5.5-beta.2",
"@mintbase-js/storage": "0.5.5-beta.2",
"@radix-ui/react-label": "^2.0.2",
"@radix-ui/react-select": "^2.0.0",
"@radix-ui/react-slot": "^1.0.2",
Expand Down
30 changes: 15 additions & 15 deletions ai-minter/pnpm-lock.yaml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Binary file added ai-minter/public/thumbnail.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file modified ai-minter/src/app/favicon.ico
Binary file not shown.
76 changes: 55 additions & 21 deletions ai-minter/src/app/layout.tsx
Original file line number Diff line number Diff line change
@@ -1,30 +1,64 @@
"use client";

import { Inter } from "next/font/google";
import { AppProvider } from "@/components/Provider";
import { Metadata } from "next";
import { headers } from "next/headers";
import "./globals.css";
import { MintbaseWalletContextProvider } from "@mintbase-js/react";
import { MintbaseWalletSetup } from "@/config/setup";
import "@near-wallet-selector/modal-ui/styles.css";
import { ReplicateProvider } from "@/providers/replicate";

const inter = Inter({ subsets: ["latin"] });
const extractSignMeta = (url: string): string | null => {
const signMetaIndex = url.indexOf("signMeta=");
if (signMetaIndex === -1) {
return null; // signMeta not found
}

const startIndex = signMetaIndex + "signMeta=".length;
const endIndex = url.indexOf("&", startIndex);
if (endIndex === -1) {
return url.substring(startIndex); // signMeta is the last parameter in the URL
} else {
return url.substring(startIndex, endIndex);
}
};

export async function generateMetadata(): Promise<Metadata> {
const headersList = headers();
const referer = headersList.get("referer");

let pageTitle = "Mintbase Minter Example";
let pageDescription =
"Learn how to Mint NFTs on NEAR with Mintbase Minter Example";

// Check if signMeta exists in the URL
const signMeta = referer ? extractSignMeta(referer) : "";
if (signMeta) {
const signMetaData = JSON.parse(decodeURIComponent(signMeta));

pageTitle = `Success! You just minted: ${signMetaData?.args?.title}`;
pageDescription = `Just Minted ${signMetaData?.args?.title} on Mintbase`;
// Now you can further process the extracted signMeta value
}

return {
metadataBase: new URL("https://ai-minter.mintbase.xyz"),
title: pageTitle,
openGraph: {
title: pageTitle,
description: pageDescription,
images: ["./thumbnail.png"],
},
twitter: {
title: pageTitle,
description: pageDescription,
siteId: "1467726470533754880",
creator: "Mintbase",
card: "summary_large_image",
images: "./thumbnail.png",
},
};
}

export default function RootLayout({
children,
}: {
children: React.ReactNode;
}) {
return (
<MintbaseWalletContextProvider {...MintbaseWalletSetup}>
<ReplicateProvider>
<html lang="en">
<body className={`${inter.className} dark`}>
<div className="flex flex-1 flex-col min-h-screen text-gray-500 gradient w-full h-full flex justify-center items-center bold text-white">
{children}
</div>
</body>
</html>
</ReplicateProvider>
</MintbaseWalletContextProvider>
);
return <AppProvider> {children} </AppProvider>;
}
47 changes: 44 additions & 3 deletions ai-minter/src/app/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,9 +5,50 @@ import { NearWalletConnector } from "@/components/NearWalletSelector";

import Head from "next/head";
import Minter from "@/components/Minter";
import { useSearchParams } from "next/navigation";
import { SuccessPage } from "@/components/Success";
import { mbUrl, nearblocksUrl } from "@/config/setup";
import { getTxnHash } from "@/hooks/utils";
import { useEffect, useState } from "react";


export default function Home() {
const { isConnected } = useMbWallet();
const [txnUrl, setTxnUrl] = useState("");


const params = useSearchParams();

const mintedParams = params.get("signMeta")
? JSON.parse(params.get("signMeta") as string)
: "";
const txnHashes = params.get("transactionHashes")
? params.get("transactionHashes")
: "";

useEffect(() => {
const fetchTxnHash = async () => {
const txn = await getTxnHash(txnHashes as string);
setTxnUrl(txn);
};

fetchTxnHash();
}, [txnHashes]);

if (mintedParams) {
const metaPage = `${mbUrl}/ref/${mintedParams.args.ref}?type=meta`;
const txnHashUrl = `${nearblocksUrl}/txns/${txnUrl}`;


const successPageData = {
nftTitle: mintedParams.args.title as string,
mediaUrl: mintedParams.args.mediaUrl as string,
metaPage,
txnHashUrl,
};

return <SuccessPage data={successPageData} />;
}

if (isConnected)
return (
Expand All @@ -22,15 +63,15 @@ export default function Home() {
<main className="flex flex-col items-center justify-center mt-2 ">
<div className="flex flex-1 flex-col w-full flex flex-col justify-center items-center space-y-8 min-h-screen text-gray-500">
<Head>
<title>Mintbase - Simple Minter Example</title>
<title>Mintbase - Simple AI Minter Example</title>
</Head>
<div className="mx-6 sm:mx-24 mt-4 mb-4">
<div className="w-full flex flex-col justify-center items-center">
<div className="w-full flex flex-col justify-center items-center space-y-8">
<div className="flex flex-col justify-center items-center space-y-8">
<h1 className="h1-90 text-5xl text-white">Mintbase Minter</h1>
<h1 className="h1-90 text-5xl text-white">Mintbase AI Minter</h1>
<h2 className="p-big-90 text-white">
A simple NFT Minter on Mintbase
A simple NFT AI Minter on Mintbase
</h2>
</div>
<div>
Expand Down
30 changes: 30 additions & 0 deletions ai-minter/src/components/Provider.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
"use client"

import { MintbaseWalletContextProvider } from "@mintbase-js/react";
import { MintbaseWalletSetup } from "@/config/setup";
import "@near-wallet-selector/modal-ui/styles.css";
import { ReplicateProvider } from "@/providers/replicate";
import { Inter } from "next/font/google";

const inter = Inter({ subsets: ["latin"] });

export const AppProvider = ({
children,
}: {
children: React.ReactNode;
}) => {

return (
<MintbaseWalletContextProvider {...MintbaseWalletSetup}>
<ReplicateProvider>
<html lang="en">
<body className={`${inter.className} dark`}>
<div className="flex flex-1 flex-col min-h-screen text-gray-500 gradient w-full h-full flex justify-center items-center bold text-white">
{children}
</div>
</body>
</html>
</ReplicateProvider>
</MintbaseWalletContextProvider>
)
}
47 changes: 47 additions & 0 deletions ai-minter/src/components/Success.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
import * as React from "react";

import { Button } from "@/components/ui/button";
import {
Card,
CardContent,
CardDescription,
CardFooter,
CardHeader,
CardTitle,
} from "@/components/ui/card";
import Link from "next/link";

interface SuccessPageData {
nftTitle: string;
mediaUrl: string;
metaPage: string;
txnHashUrl: string;
}

export function SuccessPage({ data }: { data: SuccessPageData }): JSX.Element {
const { nftTitle, mediaUrl, metaPage, txnHashUrl } = data;



return (
<Card className="w-[350px]">
<CardHeader>
<CardDescription> Success you just Minted! </CardDescription>
<CardTitle>{nftTitle}</CardTitle>
</CardHeader>
<CardContent>
<div className="flex w-full relative">
<img src={mediaUrl} width="100%" height="auto" alt={nftTitle} />
</div>
</CardContent>
<CardFooter className="flex justify-between">
<Link target="_blank" href={txnHashUrl} className="text-xs">
View Transaction
</Link>
<Link target="_blank" href={metaPage}>
<Button> View Nft on Mintbase</Button>
</Link>
</CardFooter>
</Card>
);
}
17 changes: 16 additions & 1 deletion ai-minter/src/config/setup.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,22 @@
export const proxyAddress = process?.env?.NEXT_PUBLIC_PROXY_CONTRACT_ADDRESS || "0.minsta.proxy.mintbase.testnet";
const contractAddress = process?.env?.NEXT_PUBLIC_MINT_CONTRACT_ADDRESS || "aiminter.mintspace2.testnet";
const network = process?.env?.NEXT_PUBLIC_NETWORK || "testnet";
const callbackUrl = network === "testnet" ? `https://testnet.mintbase.xyz/contract/${contractAddress}/nfts/all/0` : `https://mintbase.xyz/contract/${contractAddress}/nfts/all/0`;

const isTestnet = network === "testnet";
const callbackUrl = !isTestnet
? `https://mintbase.xyz/contract/${contractAddress}/nfts/all/0`
: `https://testnet.mintbase.xyz/contract/${contractAddress}/nfts/all/0`;
export const mbUrl = !isTestnet
? "https://www.mintbase.xyz"
: "https://testnet.mintbase.xyz";
export const nearblocksUrl = !isTestnet
? "https://nearblocks.io"
: "https://testnet.nearblocks.io";


export const nearblocksApi = !isTestnet
? "https://api.nearblocks.io"
: "https://api-testnet.nearblocks.io";

export const MintbaseWalletSetup = {
contractAddress,
Expand Down
26 changes: 24 additions & 2 deletions ai-minter/src/hooks/useMint.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import { ArweaveResponse, uploadFile, uploadReference } from "@mintbase-js/stora
import { formSchema } from "./formSchema";
import { MintbaseWalletSetup, proxyAddress } from "@/config/setup";
import { Wallet } from "@near-wallet-selector/core";
import { cbUrl } from "./utils";

const useMintImage = () => {
const { selector, activeAccountId } = useMbWallet();
Expand Down Expand Up @@ -56,7 +57,14 @@ const useMintImage = () => {
});
const file = uploadFile(media);

await handleMint(reference.id, file, activeAccountId as string, wallet);
await handleMint(
reference.id,
file,
activeAccountId as string,
wallet,
reference.media_url as string,
data.title
);
};

const form = useForm<z.infer<typeof formSchema>>({
Expand All @@ -67,12 +75,26 @@ const useMintImage = () => {
reference: string,
media: Promise<ArweaveResponse>,
activeAccountId: string,
wallet: Wallet
wallet: Wallet,
mediaUrl: string,
nftTitle: string
) {
if (reference) {

const finalMediaUrl = mediaUrl.replace("https://arweave.net/", "");

const callbackArgs = {
contractAddress: MintbaseWalletSetup.contractAddress.toString(),
amount: 1,
ref: `${reference}`,
mediaUrl: finalMediaUrl,
title: nftTitle,
};

await wallet.signAndSendTransaction({
signerId: activeAccountId,
receiverId: proxyAddress,
callbackUrl: cbUrl(reference, callbackArgs),
actions: [
{
type: "FunctionCall",
Expand Down
Loading

0 comments on commit a6f2f12

Please sign in to comment.