Cloning OpenSea NFT Boilerplate Questions

Sorry @Cipher,

Iā€™m not familiar enough with node to be able to help if you are getting the same errors with the base ethereum-boilerplateā€¦ Maybe you could start by getting help, to get the base ethereum-boilerplare running and then see if the nft markertplace can run after you have that sorted :slightly_smiling_face:

And you did everything from this post?

  1. Create a state variable called limit
  2. Passed the inputValue and limit variables to useNFTTokenIds hook
  3. Create function that increases limit by 1 when button is pressed
  4. In the useNFTTokenIds.js hook define that it takes addrs and limit as arguments
  5. Use limit as the the limit argument when calling Web3Api

yes I did all previous steps:

Blockquote
export const useNFTTokenIds = (addr,limit) => {
const { token } = useMoralisWeb3Api();
const { chainId } = useMoralisDapp();
const { resolveLink } = useIPFS();
const [NFTTokenIds, setNFTTokenIds] = useState([]);
const [totalNFTs, setTotalNFTs] = useState();
const [fetchSuccess, setFetchSuccess] = useState(true);
const {
fetch: getNFTTokenIds,
data,
error,
isLoading,
} = useMoralisWeb3ApiCall(token.getAllTokenIds, {
chain: chainId,
address: addr,
limit: 5,
//offcet: 3,
});

Blockquote
const [ limit, setLimit ] = useState(3);
const { NFTTokenIds, totalNFTs, fetchSuccess } = useNFTTokenIds(inputValue,limit);
const [visible, setVisibility] = useState(false);
const [nftToBuy, setNftToBuy] = useState(null);
const [loading, setLoading] = useState(false);
const contractProcessor = useWeb3ExecuteFunction();
const { chainId, marketAddress, contractABI, walletAddress } =
useMoralisDapp();
const nativeName = getNativeByChain(chainId);
const contractABIJson = JSON.parse(contractABI);
const { Moralis } = useMoralis();
const queryMarketItems = useMoralisQuery(ā€œMarketItemsā€);
const fetchMarketItems = JSON.parse(
JSON.stringify(queryMarketItems.data, [
ā€œobjectIdā€,
ā€œcreatedAtā€,
ā€œpriceā€,
ā€œnftContractā€,
ā€œitemIdā€,
ā€œsoldā€,
ā€œtokenIdā€,
ā€œsellerā€,
ā€œownerā€,
ā€œconfirmedā€,
])
);
const purchaseItemFunction = ā€œcreateMarketSaleā€;
const NFTCollections = getCollectionsByChain(chainId);

function moreNFTs() {
setLimit(limit + 1)
} //make sure you call this function onClick of button

1 Like

limit needs to be ā€œlimitā€ here not 5. Other wise you are just fetching the same 5 NFTs every time

You are legend :slight_smile:
Thank you very very very much, it works now.
but when I press View Collection always appear only 3 NFT, when press button it load (limit+ā€¦)any number that I put here, but I am wondering why always start with only 3 NFT ? this is not important for me as long as it works fine.
I guess every time we press the button the limit will increase (accumulate) I realised that lets say I put: setLimit(limit + 100), it will load normally but after few pressed it start to take ages because I guess limit increase every time we pressed? is there anyway to clear the limit after every load?
Now I will deploy the Dapp and hopefully works fine,
other question if you donā€™t mind can we use a custom host name for the Dapp or should be with Moralis? do you have any service for that (paid monthly or so) last but not least how can I change the logo ? is there any particular place in the code we can change?
Thanks mate for your help and quick response I really appreciate it.

Omg @IAmJaysWay, you are awesome! I got every transaction now! Thank you so muuch, my ABI event name was wrong all the way lol.

About the list() function, it doesn`t exists, only succlist, wich is a modal opening function.

1 Like

Hi @IAmJaysWay

Yes that true I was started from scratch and setAprovalFroAll was missing, Thanks !! the issue is solved and the contract is working as expected.

Best Regards

1 Like

@engg you can change the number of initially loaded NFTs over here:

and yes although this works, it would be better practice to store the NFTs in some state variable and every time you press your button you change the offset parameter to be passed to the useNFTTokenIds() hook, which just gets the next set of NFTs and then concatenate these to your NFTs state variable. I trust you can figure this out :wink:

As for deploying you can check out this video, https://www.youtube.com/watch?v=RiH0ADGPleg or then host it yourself.

The Logo is a Moralis SVG, but you can import an image if you like and replace the <Logo /> in App.jsx

.
.
<Header style={styles.header}>
  <Logo />                                            // Your Logo here can be an <img>
  <SearchCollections setInputValue={setInputValue}/>
.
.

Thank you very much
I will do it
I wish you and all members here have a great :+1: year a head :pray::rose:

1 Like

the transaction fails because it needs a special little function. took me litterally 6 hours to find. hope this helps someone
event Received(address, uint);
receive() external payable {
emit Received(msg.sender, msg.value);
}

1 Like

Hello All,
Regarding to marketplaceBoilerplate.sol if I donā€™t change anything in the contract and deploy it to any network, my question is that I guess the contractABI is still the same (no change!) the only thing that will change is the address? (marketAddress), Am I right?
Also the listener in the Server will stay as it is, if I donā€™t change anything in the contract for new deploying? So if I deploy to any network, all stay as it is but the only thing will change is the new address?

Yes, this should be right.

Thanks malik :slight_smile:
What about the server side for listener and synchronize? I guess will stay as it is even if I deploy a new contract without any changing?

If you are referring to syncing the smart contract events on Moralis, just make sure you select the correct chain and provide the correct contract address (of the newly deployed contract). otherwise the details should be the same :+1:

this cloned version of opensea. am I able to build off it to add additional features etc?

Ofc @heyimsurge, feel free to fork the repo and add/modify/delete features to your own liking :slight_smile:

i saw that as well. Having a terrible issue with the address that remix deploys to

Hello me again :slight_smile:
I follow the link of deploying the Dapp, it went through all steps correctly and finally in the last step I got error of following question
What server do you want to connect to?:
I only have one server and try 1, 2, 3ā€¦ all show Invalid input! till I realized that is 0, it works but through an error:
En error occured when trying to deploy
Error: Request failed with status code 502
at createError (C:\Users\user\AppData\Roaming\npm\node_modules\moralis-admin-cli\node_modules\axios\lib\core\createError.js:16:15)
at settle (C:\Users\user\AppData\Roaming\npm\node_modules\moralis-admin-cli\node_modules\axios\lib\core\settle.js:17:12)
at IncomingMessage.handleStreamEnd (C:\Users\user\AppData\Roaming\npm\node_modules\moralis-admin-cli\node_modules\axios\lib\adapters\http.js:269:11)
at IncomingMessage.emit (node:events:402:35)
at IncomingMessage.emit (node:domain:475:12)
at endReadableNT (node:internal/streams/readable:1343:12)
at processTicksAndRejections (node:internal/process/task_queues:83:21) {
config: {
.
.
.
long config here
.
.
.
},
isAxiosError: true,
toJSON: [Function: toJSON]
}

this where I deploy in the folder of the project
C:\Users\userā€¦\public>moralis-admin-cli deploy

Okay, Iā€™m having some trouble again.
Iā€™ve successfully deployed the marketplace contract to Mumbai testnet and was able to interact with it as I wanted.

However, now when I deployed the same contract to Ethereum mainnet I have some issues.
The NFT balance is successfully shown but once I list the NFTā€™s for sale it does not show the transactions under the transactions tab nor does the explore page update the NFT to the green ā€œBuy nowā€ display.
I had the exact same problem earlier but fixed it by changing:

const queryMarketItems = useMoralisQuery(ā€œMarketItemsā€);
.
.
.
const marketList = Moralis.Object.extend("MarketItems");

in NFTTokenIds.jsx and NFTMarketTransactions.jsx to:

const queryMarketItems = useMoralisQuery(ā€œCreatedMarketItemsā€);
.
.
.
const marketList = Moralis.Object.extend("CreatedMarketItems");

This worked like a charm while I used the Mumbai testnet, but now I have a similar issue despite using the same code.
The database class CreatedMarketItems in my Ethereum mainnet dApp is populated successfully and have the exact same structure as the one I used for Mumbai testnet.
From my understanding there has to be some query which is faulty, but where?

This is my complete NFTTokenIds.jsx:

import React, { useState, useEffect } from "react";
import { getNativeByChain } from "helpers/networks";
import { getCollectionsByChain } from "helpers/collections";
import {
  useMoralis,
  useMoralisQuery,
  useNewMoralisObject,
} from "react-moralis";
import { Card, Image, Tooltip, Modal, Badge, Alert, Spin } from "antd";
import { useNFTTokenIds } from "hooks/useNFTTokenIds";
import {
  FileSearchOutlined,
  RightCircleOutlined,
  ShoppingCartOutlined,
} from "@ant-design/icons";
import { useMoralisDapp } from "providers/MoralisDappProvider/MoralisDappProvider";
import { getExplorer } from "helpers/networks";
import { useWeb3ExecuteFunction } from "react-moralis";
const { Meta } = Card;

const styles = {
  NFTs: {
    display: "flex",
    flexWrap: "wrap",
    WebkitBoxPack: "start",
    justifyContent: "flex-start",
    margin: "0 auto",
    maxWidth: "1000px",
    gap: "10px",
  },
  banner: {
    display: "flex",
    justifyContent: "space-evenly",
    alignItems: "center",
    margin: "0 auto",
    width: "600px",
    //borderRadius: "10px",
    height: "150px",
    marginBottom: "40px",
    paddingBottom: "20px",
    borderBottom: "solid 1px #e3e3e3",
  },
  logo: {
    height: "115px",
    width: "115px",
    borderRadius: "50%",
    // positon: "relative",
    // marginTop: "-80px",
    border: "solid 4px white",
  },
  text: {
    color: "#041836",
    fontSize: "27px",
    fontWeight: "bold",
  },
};

function NFTTokenIds({ inputValue, setInputValue }) {
  const fallbackImg =
    "";
  const { NFTTokenIds, totalNFTs, fetchSuccess } = useNFTTokenIds(inputValue);
  const [visible, setVisibility] = useState(false);
  const [nftToBuy, setNftToBuy] = useState(null);
  const [loading, setLoading] = useState(false);
  const contractProcessor = useWeb3ExecuteFunction();
  const { chainId, marketAddress, contractABI, walletAddress } =
    useMoralisDapp();
  const nativeName = getNativeByChain(chainId);
  const contractABIJson = JSON.parse(contractABI);
  const { Moralis } = useMoralis();
  const queryMarketItems = useMoralisQuery("CreatedMarketItems");
  const fetchMarketItems = JSON.parse(
    JSON.stringify(queryMarketItems.data, [
      "objectId",
      "createdAt",
      "price",
      "nftContract",
      "itemId",
      "sold",
      "tokenId",
      "seller",
      "owner",
      "confirmed",
    ])
  );
  const purchaseItemFunction = "createMarketSale";
  const NFTCollections = getCollectionsByChain(chainId);

  async function purchase() {
    setLoading(true);
    const tokenDetails = getMarketItem(nftToBuy);
    const itemID = tokenDetails.itemId;
    const tokenPrice = tokenDetails.price;
    const ops = {
      contractAddress: marketAddress,
      functionName: purchaseItemFunction,
      abi: contractABIJson,
      params: {
        nftContract: nftToBuy.token_address,
        itemId: itemID,
      },
      msgValue: tokenPrice,
    };

    await contractProcessor.fetch({
      params: ops,
      onSuccess: () => {
        console.log("success");
        setLoading(false);
        setVisibility(false);
        updateSoldMarketItem();
        succPurchase();
      },
      onError: (error) => {
        setLoading(false);
        failPurchase();
      },
    });
  }

  const handleBuyClick = (nft) => {
    setNftToBuy(nft);
    console.log(nft.image);
    setVisibility(true);
  };

  function succPurchase() {
    let secondsToGo = 5;
    const modal = Modal.success({
      title: "Success!",
      content: `You have purchased this NFT`,
    });
    setTimeout(() => {
      modal.destroy();
    }, secondsToGo * 1000);
  }

  function failPurchase() {
    let secondsToGo = 5;
    const modal = Modal.error({
      title: "Error!",
      content: `There was a problem when purchasing this NFT`,
    });
    setTimeout(() => {
      modal.destroy();
    }, secondsToGo * 1000);
  }

  async function updateSoldMarketItem() {
    const id = getMarketItem(nftToBuy).objectId;
    const marketList = Moralis.Object.extend("CreatedMarketItems");
    const query = new Moralis.Query(marketList);
    await query.get(id).then((obj) => {
      obj.set("sold", true);
      obj.set("owner", walletAddress);
      obj.save();
    });
  }

  const getMarketItem = (nft) => {
    const result = fetchMarketItems?.find(
      (e) =>
        e.nftContract === nft?.token_address &&
        e.tokenId === nft?.token_id &&
        e.sold === false &&
        e.confirmed === true
    );
    return result;
  };

  return (
    <>
      <div>
        {contractABIJson.noContractDeployed && (
          <>
            <Alert
              message="No Smart Contract Details Provided. Please deploy smart contract and provide address + ABI in the MoralisDappProvider.js file"
              type="error"
            />
            <div style={{ marginBottom: "10px" }}></div>
          </>
        )}
        {inputValue !== "explore" && totalNFTs !== undefined && (
          <>
            {!fetchSuccess && (
              <>
                <Alert
                  message="Unable to fetch all NFT metadata... We are searching for a solution, please try again later!"
                  type="warning"
                />
                <div style={{ marginBottom: "10px" }}></div>
              </>
            )}
            <div style={styles.banner}>
              <Image
                preview={false}
                src={NFTTokenIds[0]?.image || "error"}
                fallback={fallbackImg}
                alt=""
                style={styles.logo}
              />
              <div style={styles.text}>
                <>
                  <div>{`${NFTTokenIds[0]?.name}`}</div>
                  <div
                    style={{
                      fontSize: "15px",
                      color: "#9c9c9c",
                      fontWeight: "normal",
                    }}
                  >
                    Collection Size: {`${totalNFTs}`}
                  </div>
                </>
              </div>
            </div>
          </>
        )}

        <div style={styles.NFTs}>
          {inputValue === "explore" &&
            NFTCollections?.map((nft, index) => (
			<Card
                hoverable
                actions={[
                  <Tooltip title="View Collection">
                    <RightCircleOutlined
                      onClick={() => setInputValue(nft?.addrs)}
                    />
                  </Tooltip>,
                ]}
                style={{ width: 240, border: "2px solid #e7eaf3" }}
                cover={
                  <Image
                    preview={false}
                    src={nft?.image || "error"}
                    fallback={fallbackImg}
                    alt=""
                    style={{ height: "240px" }}
                  />
                }
                key={index}
              >
                <Meta title={nft.name} />
              </Card>
            ))}

          {inputValue !== "explore" &&
            NFTTokenIds.slice(0, 20).map((nft, index) => (
              <Card
                hoverable
                actions={[
                  <Tooltip title="View On Blockexplorer">
                    <FileSearchOutlined
                      onClick={() =>
                        window.open(
                          `${getExplorer(chainId)}address/${nft.token_address}`,
                          "_blank"
                        )
                      }
                    />
                  </Tooltip>,
                  <Tooltip title="Buy NFT">
                    <ShoppingCartOutlined onClick={() => handleBuyClick(nft)} />
                  </Tooltip>,
                ]}
                style={{ width: 240, border: "2px solid #e7eaf3" }}
                cover={
                  <Image
                    preview={false}
                    src={nft.image || "error"}
                    fallback={fallbackImg}
                    alt=""
                    style={{ height: "240px" }}
                  />
                }
                key={index}
              >
                {getMarketItem(nft) && (
                  <Badge.Ribbon text="Buy Now" color="green"></Badge.Ribbon>
                )}
                <Meta title={nft.name} description={`#${nft.token_id}`} />
              </Card>
            ))}
        </div>
        {getMarketItem(nftToBuy) ? (
          <Modal
            title={`Buy ${nftToBuy?.name} #${nftToBuy?.token_id}`}
            visible={visible}
            onCancel={() => setVisibility(false)}
            onOk={() => purchase()}
            okText="Buy"
          >
            <Spin spinning={loading}>
              <div
                style={{
                  width: "250px",
                  margin: "auto",
                }}
              >
                <Badge.Ribbon
                  color="green"
                  text={`${
                    getMarketItem(nftToBuy).price / ("1e" + 18)
                  } ${nativeName}`}
                >
                  <img
                    src={nftToBuy?.image}
                    style={{
                      width: "250px",
                      borderRadius: "10px",
                      marginBottom: "15px",
                    }}
                  />
                </Badge.Ribbon>
              </div>
            </Spin>
          </Modal>
        ) : (
          <Modal
            title={`Buy ${nftToBuy?.name} #${nftToBuy?.token_id}`}
            visible={visible}
            onCancel={() => setVisibility(false)}
            onOk={() => setVisibility(false)}
          >
            <img
              src={nftToBuy?.image}
              style={{
                width: "250px",
                margin: "auto",
                borderRadius: "10px",
                marginBottom: "15px",
              }}
            />
            <Alert
              message="This NFT is currently not for sale"
              type="warning"
            />
          </Modal>
        )}
      </div>
    </>
  );
}

export default NFTTokenIds;

and NFTMarketTransactions.jsx:

import React, { useState } from "react";
import { useMoralis, useMoralisQuery } from "react-moralis";
import { useMoralisDapp } from "providers/MoralisDappProvider/MoralisDappProvider";
import { Table, Tag, Space } from "antd";
import { PolygonCurrency} from "./Chains/Logos";
import moment from "moment";

const styles = {
  table: {
    margin: "0 auto",
    width: "1000px",
  },
};

function NFTMarketTransactions() {
  const { walletAddress } = useMoralisDapp();
  const { Moralis } = useMoralis();
  const queryItemImages = useMoralisQuery("ItemImages");
  const fetchItemImages = JSON.parse(
    JSON.stringify(queryItemImages.data, [
      "nftContract",
      "tokenId",
      "name",
      "image",
    ])
  );
  const queryMarketItems = useMoralisQuery("CreatedMarketItems");
  const fetchMarketItems = JSON.parse(
    JSON.stringify(queryMarketItems.data, [
      "updatedAt",
      "price",
      "nftContract",
      "itemId",
      "sold",
      "tokenId",
      "seller",
      "owner",
    ])
  )
    .filter(
      (item) => item.seller === walletAddress || item.owner === walletAddress
    )
    .sort((a, b) =>
      a.updatedAt < b.updatedAt ? 1 : b.updatedAt < a.updatedAt ? -1 : 0
    );

  function getImage(addrs, id) {
    const img = fetchItemImages.find(
      (element) =>
        element.nftContract === addrs &&
        element.tokenId === id
    );
    return img?.image;
  }

  function getName(addrs, id) {
    const nme = fetchItemImages.find(
      (element) =>
        element.nftContract === addrs &&
        element.tokenId === id
    );
    return nme?.name;
  }



  const columns = [
    {
      title: "Date",
      dataIndex: "date",
      key: "date",
    },
    {
      title: "Item",
      key: "item",
      render: (text, record) => (
        <Space size="middle">
          <img src={getImage(record.collection, record.item)} style={{ width: "40px", borderRadius:"4px"}} />
          <span>#{record.item}</span>
        </Space>
      ),
    },
    {
      title: "Collection",
      key: "collection",
      render: (text, record) => (
        <Space size="middle">
          <span>{getName(record.collection, record.item)}</span>
        </Space>
      ),
    },
    {
      title: "Transaction Status",
      key: "tags",
      dataIndex: "tags",
      render: (tags) => (
        <>
          {tags.map((tag) => {
            let color = "geekblue";
            let status = "BUY";
            if (tag === false) {
              color = "volcano";
              status = "waiting";
            } else if (tag === true) {
              color = "green";
              status = "confirmed";
            }
            if (tag === walletAddress) {
              status = "SELL";
            }
            return (
              <Tag color={color} key={tag}>
                {status.toUpperCase()}
              </Tag>
            );
          })}
        </>
      ),
    },
    {
      title: "Price",
      key: "price",
      dataIndex: "price",
      render: (e) => (
        <Space size="middle">
          <PolygonCurrency/>
          <span>{e}</span>
        </Space>
      ),
    }
  ];

  const data = fetchMarketItems?.map((item, index) => ({
    key: index,
    date: moment(item.updatedAt).format("DD-MM-YYYY HH:mm"),
    collection: item.nftContract,
    item: item.tokenId,
    tags: [item.seller, item.sold],
    price: item.price / ("1e" + 18)
  }));

  return (
    <>
      <div>
        <div style={styles.table}>
          <Table columns={columns} dataSource={data} />
        </div>
      </div>
    </>
  );
}

export default NFTMarketTransactions;
const columns = [
  {
    title: "Date",
    dataIndex: "date",
    key: "date",
  },
  {
    title: "Item",
    key: "item",

  },
  {
    title: "Collection",
    key: "collection",
  },
  {
    title: "Transaction Status",
    key: "tags",
    dataIndex: "tags",
  },
  {
    title: "Price",
    key: "price",
    dataIndex: "price",
  }
];

I canā€™t find anything different besides that my Ethereum mainnet dApp database is only populated with 4 classes while the Mumbai testnet database was automatically populated with many more classes, i.e:

ItemImages
PolygonBalance
PolygonBalancePending
PolygonNFTOwners
PolygonNFTOwnersPending
PolygonNFTTransfers
PolygonTransactions

I see that ItemImages is queried in NFTMarketTransactions.jsx.
This is the only thing I can come up with but donā€™t know how I should solve it.
Iā€™ve made some minor aesthetic changes like removing the text in the footer.
I also gave the deployed Ethereum mainnet contract another name.
But I donā€™t see how any of these changes could cause my current error.

Thanks!

Hi @63MoneyT,

Is it possible to github commit / push (under a new name?) your version so I and others (as @latok) can see your improvements, compare and learn from it?
Should be great!
Thank you :wink: !