//SPDX-License-Identifier: MIT
/*@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@ @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@*/
/**
* @title Auction
* @author @brougkr
* @notice A Smart Contract To Facilitate Ascending Rebate Auctions (With Ascending Rebate Reserve Floor) For Multiple NFTs (Or Whatever Else You Want To Sell)
*/
pragma solidity 0.8.19;
import { DelegateCashEnabled } from "./DelegateCashEnabled.sol";
import { ReentrancyGuard } from "@openzeppelin/contracts/security/ReentrancyGuard.sol";
import { Ownable } from "@openzeppelin/contracts/access/Ownable.sol";
import { MerkleProof } from "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol";
contract AuctionMarketplace is DelegateCashEnabled, ReentrancyGuard, Ownable
{
struct Bid
{
uint _OGBidIndex; // [0] -> Original Bid Index
uint _Priority; // [1] -> Priority Of Bid (Merkle Rank)
uint _Allocation; // [2] -> Allocation For Provided Priority Tier
uint _ETHValue; // [3] -> ETH Value Of Bid
uint _Timestamp; // [4] -> Unix Timestamp Of Bid Confirmation
address _Bidder; // [5] -> Wallet Address Of Bidder
address _Vault; // [6] -> Wallet Address Of Vault (optional `delegate.cash` support)
bool _Rebated; // [7] -> If Bidder Rebated ETH From Bid
bool _Winner; // [8] -> If Bidder Is A Winner (Top Placing) In The Auction
bool _NFTSent; // [9] -> If Bidder Has Received Their NFT
}
struct Params
{
string _Name; // [0] -> Name Of Auction
bool _SettlementEnabled; // [1] -> If Rebate Last Price Is Enabled (Everyone Pays Lowest Leaderboard Price)
bool _UserSettlementEnabled; // [2] -> If Self-Service User Settlement Is Enabled (Bidders Can Settle Their Own Bids ETH & NFTs)
uint _LeaderboardSize; // [3] -> The Bid Threshold For NewMinimumBid (eg. 50 Valid Bids)
uint _UnixStartTime; // [4] -> Unix Start Time Of Auction
uint _UnixEndTime; // [5] -> Unix End Time Of Auction
uint _MinBIPSIncrease; // [6] -> Minimum BIPS (%) Increase On Each Subsequent Bid After Configured LeaderboardSize
uint _SecondsExtension; // [7] -> # Of Second(s) Of Extension For Auction (Input In Seconds)
uint _SecondsThreshold; // [8] -> # Of Seconds Within Auction End Time To Be Eligible For Auction Extension
uint _InitialMinimumBid; // [9] -> Initial Minimum Bid
uint _ProjectIDMintPass; // [10] -> The Factory MintPass ProjectID
uint _MintPackAmount; // [11] -> The Amount Of Mint Passes To Disperse Per Bid
address _NFT; // [12] -> Address Of NFT Contract
address _Operator; // [13] -> Wallet Holding NFTs To Disperse
}
struct State
{
bool _Active; // [0] -> _Active
bool _NFTsDispersed; // [1] -> _NFTsDispersed
uint _LastMinBid; // [2] -> _LastMinBid
uint _GlobalUniqueBids; // [3] -> _GlobalUniqueBids
}
struct AllAuctionParams
{
string _Name; // [0] -> Name Of Auction
bool _Active; // [1] -> If Sale Is Active
bool _NFTsDispersed; // [2] -> If NFTs Have Been Dispersed
bool _SettlementEnabled; // [3] -> If Rebate Last Price Is Enabled (Everyone Pays Lowest Leaderboard Price)
uint _LeaderboardSize; // [4] -> The Bid Threshold For NewMinimumBid (eg. 50 Valid Bids)
uint _UnixStartTime; // [5] -> Unix Start Time Of Auction
uint _UnixEndTime; // [6] -> Unix End Time Of Auction
uint _MinBIPSIncrease; // [7] -> Minimum BIPS (%) Increase On Each Subsequent Bid After 50 Unique Bids
uint _SecondsExtension; // [8] -> # Of Seconds(s) Of Extension For Auction (Input In # Of Seconds)
uint _SecondsThreshold; // [9] -> # Of Seconds Within Auction End Time To Be Eligible For Auction Extension
uint _LastMinBid; // [10] -> Value Of The Last Minimum Bid
uint _GlobalUniqueBids; // [11] -> # Of Global Unique Bids
uint _ProjectIDMintPass; // [12] -> The Factory MintPass ProjectID
uint _MintPackAmount; // [13] -> The Amount Of Mint Passes To Disperse Per Bid
address _NFT; // [14] -> Address Of NFT Contract
}
/*-----------------------------
* STATE VARIABLES & MAPPINGS *
------------------------------*/
mapping(uint=>Params) public AuctionParams;
mapping(uint=>State) public SaleState;
mapping(uint=>mapping(uint=>Bid)) public Bids;
mapping(uint=>mapping(uint=>uint)) public Leaderboard;
mapping(uint=>mapping(address=>uint[])) public UserBidIndexes;
mapping(address=>bool) public Admin;
mapping(uint=>mapping(uint=>bool)) public NFTTokenIDHasBeenSent;
mapping(uint=>uint[]) public Discounts;
mapping(uint=>bytes32[]) public MerkleRootsEligibility;
mapping(uint=>bytes32[]) public MerkleRootsAmounts;
mapping(uint=>mapping(address=>mapping(uint=>uint))) public PriorityPurchaseAmount; // [SaleIndex][Wallet][Priority] => Purchased Amount For Priority Level
address private constant _BRT_MULTISIG = 0x0BC56e3c1397e4570069e89C07936A5c6020e3BE;
uint public _GLOBAL_UNIQUE_SALES;
/*---------
* EVENTS *
----------*/
/**
* @dev Emitted When A New Bid Is Submitted
*/
event NewBidComplete(uint BidIndex, address Bidder, uint MessageValue, uint Unixtimestamp, address Vault);
/**
* @dev Emitted When A Bid Is Topped Up
*/
event BidToppedUp(uint BidIndex, uint ETHForBid, uint Unixtimestamp, address Bidder);
/**
* @dev Emitted When A Bid Reclaim Fails
*/
event BidReclaimFailed(uint BidIndex);
/**
* @dev Emitted When A Bid Reclaim Succeeds
*/
event BidReclaimSuccess(uint BidIndex);
/**
* @dev Emitted When A Bidder's ETH Is Rebated (The Bid They Are Trying To Top Up Was Frontran)
*/
event BidTopupRefunded(uint Rebate);
/**
* @dev Emitted When The Auction End Time Is Extended
*/
event AuctionExtended();
/**
* @dev Emitted When A Bid Is Refunded (Kicked Back To Losing Bidder)
*/
event BidRefunded(uint BidIndex);
/*--------------
* CONSTRUCTOR *
---------------*/
constructor()
{
Admin[msg.sender] = true; // sets owner as admin
Admin[0xe06F5FAE754e81Bc050215fF89B03d9e9FF20700] = true;
Admin[0x90D98d5A1fD21B7cEa4D5c18341607ed1a8345c0] = true;
Admin[0x18B7511938FBe2EE08ADf3d4A24edB00A5C9B783] = true;
SaleState[0]._Active = true; // activates auction
SaleState[0]._LastMinBid = 1 ether; // starts auction specified ETH value
AuctionParams[0]._Name = 'Buenos Aires Artist Mint Packs'; // sets auction name
AuctionParams[0]._SettlementEnabled = true; // enables rebate last price
// AuctionParams[0]._UserSettlementEnabled = false; // enables self-service user auction settlement
AuctionParams[0]._MintPackAmount = 10; // sets mint pass transfer amount
AuctionParams[0]._LeaderboardSize = 25; // (max # of bids on leaderboard)
AuctionParams[0]._UnixStartTime = block.timestamp; // sets auction start time
AuctionParams[0]._UnixEndTime = block.timestamp + 694200 seconds; // sets auction end time
AuctionParams[0]._MinBIPSIncrease = 105; // 5% Increase On Each Subsequent Bid After 50 Unique Bids
AuctionParams[0]._SecondsExtension = 300 seconds; // # Of Seconds Of Extension
AuctionParams[0]._SecondsThreshold = 300 seconds; // # Of Seconds Within Auction End Time Where Auction Extension Is Enabled
AuctionParams[0]._Operator = msg.sender; // sets operator
MerkleRootsEligibility[0].push(0xbb9d32f908b1b7ec4241468d3ec82305ac46bddc361b4da9a500625a5bed5986); // Merkle Root Priority Eligibility Full Set
MerkleRootsEligibility[0].push(0xbb9d32f908b1b7ec4241468d3ec82305ac46bddc361b4da9a500625a5bed5986); // Merkle Root Priority Eligibility Citizen
MerkleRootsAmounts[0].push(0xbb9d32f908b1b7ec4241468d3ec82305ac46bddc361b4da9a500625a5bed5986); // Merkle Root Priority Allocation Full Set
MerkleRootsAmounts[0].push(0xbb9d32f908b1b7ec4241468d3ec82305ac46bddc361b4da9a500625a5bed5986); // Merkle Root Priority Allocation Citizen
Discounts[0].push(80); // 20% Discount
Discounts[0].push(90); // 10% Discount
}
/*-----------------
* USER FUNCTIONS *
------------------*/
/**
* @dev Submits A New Bid To The Auction
* @param SaleIndex The Sale Index To Target
* @param MaxAmount The Maximum Merkle Priority Allocation Amount | note: Input 0 If Not Applicable
* @param Vault Optional delegate.xyz Integration | note: Input 0x0000000000000000000000000000000000000000 If No Delegate
* @param ProofEligibility The Merkle Proof For The Bidder's Priority | note: Input [0x0000000000000000000000000000000000000000000000000000000000000000] If No Merkle Proof
* @param ProofAmount The Merkle Proof For The Bidder's Priority Allocation | note: Input [0x0000000000000000000000000000000000000000000000000000000000000000] If No Merkle Proof
*/
function NewBid ( uint SaleIndex, uint MaxAmount, address Vault, bytes32[] calldata ProofEligibility, bytes32[] calldata ProofAmount ) external payable nonReentrant
{
require(tx.origin == msg.sender, "Auction: EOA Only, Use `delegate.cash` For Wallet Delegation"); // Requires `msg.sender` Is A Valid EOA
require(SaleState[SaleIndex]._Active, "Auction: Auction Has Ended"); // Requires The Auction Is Active
require(block.timestamp >= AuctionParams[SaleIndex]._UnixStartTime, "Auction: Auction Has Not Started"); // Requires The Auction Has Started
require(block.timestamp < AuctionParams[SaleIndex]._UnixEndTime, "Auction: Auction Has Concluded"); // Requires The Auction Has Not Ended
address Recipient = msg.sender;
if(Vault != address(0)) { if(DelegateCash.checkDelegateForAll(msg.sender, Vault)) { Recipient = Vault; } } // `delegate.cash` Integration
__FinalizeNewBid(SaleIndex, msg.value); // Auto-Calculates The Required Reserve Price For The Bid (+5%)
__CheckAndSeedAuctionEndTime(SaleIndex); // Checks If Auction End Time Should Be Extended And Extends If Necessary
uint Priority = _ValidateMerkleProofs(SaleIndex, MaxAmount, Recipient, ProofEligibility, ProofAmount); // Validates Merkle Proof And Returns Merkle Priority
uint CurrentBidIndex = SaleState[SaleIndex]._GlobalUniqueBids; // Current Bid Index
Bids[SaleIndex][CurrentBidIndex] = Bid (
CurrentBidIndex, // [0] -> Original Bid Index
Priority, // [1] -> Priority Of Bid (Merkle Rank)
MaxAmount, // [2] -> Allocation For Provided Priority Tier
msg.value, // [3] -> ETH Value Of Bid
block.timestamp, // [4] -> Unix Timestamp Of Bid Confirmation
msg.sender, // [5] -> Wallet Address Of Bidder
Recipient, // [6] -> Wallet Address Of Vault (optional `delegate.cash` support)
false, // [7] -> If Bidder Rebated ETH From Bid
false, // [8] -> If Bidder Is A Winner (Top Placing) In The Auction
false // [9] -> If Bidder Has Received Their NFT
); // Registers New Bid
UserBidIndexes[SaleIndex][msg.sender].push(CurrentBidIndex); // Appends Bid Index To User's Bid Indexes
SaleState[SaleIndex]._GlobalUniqueBids = CurrentBidIndex + 1; // Increments Global Unique Bids
emit NewBidComplete(CurrentBidIndex, msg.sender, msg.value, block.timestamp, Recipient); // Emits Bid Event
}
/**
* @dev Tops Up Bid(s) With Additional ETH
* @param SaleIndex The Sale Index To Target
* @param BidIndexes[] The Bid Indexes To Top Up
* @param Amounts[] The Amounts (In WEI) To Top Up The Corresponding Bid Indexes By
*/
function IncreaseBid ( uint SaleIndex, uint[] calldata BidIndexes, uint[] calldata Amounts ) external payable nonReentrant
{
require(tx.origin == msg.sender, "Auction: EOA Only, Use `delegate.cash` For Wallet Delegation"); // Requires `msg.sender` Is A Valid EOA
require(SaleState[SaleIndex]._Active, "Auction: Auction Has Ended"); // Requires The Auction Is Active
require(block.timestamp >= AuctionParams[SaleIndex]._UnixStartTime, "Auction: Auction Has Not Started"); // Requires The Auction Has Started
require(block.timestamp < AuctionParams[SaleIndex]._UnixEndTime, "Auction: Auction Has Concluded"); // Requires The Auction Has Not Ended
require(BidIndexes.length == Amounts.length, "Auction: BidIndexes And Amounts Array Length Mismatch"); // Requires BidIndexes And Amounts Length Match
require(BidIndexes.length > 0, "Auction: User Has Input No Bids To Top Up"); // Requires User Has Bids To Top Up
if(!AuctionParams[SaleIndex]._SettlementEnabled) { __CheckAndSeedAuctionEndTime(SaleIndex); } // Checks If Auction End Time Should Be Extended And Extends If Necessary
Bid memory _Bid;
uint Total;
uint BidValue;
for(uint x; x < BidIndexes.length; x++)
{
_Bid = Bids[SaleIndex][BidIndexes[x]];
BidValue = _Bid._ETHValue;
if(!_Bid._Rebated && !_Bid._Winner)
{
require(msg.sender == _Bid._Bidder, "Auction: `msg.sender` Is Not The Bidder Of Desired Bid Index");
require(BidValue + Amounts[x] >= (BidValue * AuctionParams[SaleIndex]._MinBIPSIncrease) / 100, "Auction: Bid Amount Topup Requires >= 5% Increase");
Bids[SaleIndex][BidIndexes[x]]._ETHValue += Amounts[x];
Bids[SaleIndex][BidIndexes[x]]._Timestamp = block.timestamp;
Total += Amounts[x];
emit BidToppedUp(BidIndexes[x], Amounts[x], block.timestamp, msg.sender);
}
}
require(msg.value >= Total, "Auction: Invalid ETH Values");
uint Rebate = msg.value - Total; // Rebates Excess ETH (If Total > msg.value This Will Revert)
if(Rebate > 0)
{
(bool Success, ) = msg.sender.call { value: Rebate }("");
require(Success, "Auction: Failed To Rebate Excess ETH To Bidder, Resubmit Transaction");
emit BidTopupRefunded(Rebate);
}
}
/**
* @dev Rebates ETH From Bid(s) If Bidder Is Not A Winner & Disperses NFTs If Winner
* @param SaleIndex The Sale Index To Finalize
*/
function UserSettleAuction ( uint SaleIndex ) external nonReentrant
{
require(tx.origin == msg.sender, "Auction: EOA Only, Use `delegate.cash` For Wallet Delegation"); // Requires `msg.sender` Is A Valid EOA
require(!SaleState[SaleIndex]._NFTsDispersed, "Auction: NFTs Have Been Dispersed");
require(AuctionParams[SaleIndex]._UserSettlementEnabled, "Auction: User Settlement Is Not Enabled");
require(UserBidIndexes[SaleIndex][msg.sender].length > 0, "Auction: User Has No Bids To Settle");
require(block.timestamp > AuctionParams[SaleIndex]._UnixEndTime, "Auction: Cannot Finalize, Auction Is Still Active");
address Bidder = msg.sender;
__UserDisperseETH(SaleIndex, Bidder);
__UserDisperseNFT(SaleIndex, Bidder);
}
/*------------------
* ADMIN FUNCTIONS *
-------------------*/
/**
* @dev Starts Auction
* @param AuctionInfo The Struct Of Auction Info
* @param RootsEligibility The Merkle Roots For Priority Eligibility
* @param RootsAmounts The Merkle Roots For Priority Allocation
* @param DiscountAmounts The Discount Amounts For The Auction ([80,90] = 20% Discount, 10% Discount)
*/
function __StartAuction (
Params memory AuctionInfo,
bytes32[] calldata RootsEligibility,
bytes32[] calldata RootsAmounts,
uint[] calldata DiscountAmounts
) external onlyAdmin returns (uint SaleIndex) {
require(AuctionInfo._UnixStartTime > block.timestamp, "Auction: Start Time Must Be In The Future");
require(RootsEligibility.length == RootsAmounts.length && RootsAmounts.length == DiscountAmounts.length, "Auction: Array Lengths Must Match");
SaleIndex = _GLOBAL_UNIQUE_SALES;
AuctionParams[SaleIndex] = AuctionInfo;
SaleState[SaleIndex]._LastMinBid = AuctionParams[SaleIndex]._InitialMinimumBid;
Discounts[SaleIndex] = DiscountAmounts;
MerkleRootsEligibility[SaleIndex] = RootsEligibility;
MerkleRootsAmounts[SaleIndex] = RootsAmounts;
_GLOBAL_UNIQUE_SALES = SaleIndex + 1;
for(uint x; x < DiscountAmounts.length; x++) { require(DiscountAmounts[x] <= 100, "Invalid Discount Amount"); }
return SaleIndex;
}
/**
* @dev Changes The Sale Roots
* @param SaleIndex The Sale Index To Change
* @param RootsEligibility The New Merkle Roots For Priority Eligibility
* @param RootsAmounts The New Merkle Roots For Max Priority Allocation
*/
function ___ChangeRoots(uint SaleIndex, bytes32[] calldata RootsEligibility, bytes32[] calldata RootsAmounts) external onlyAdmin
{
MerkleRootsEligibility[SaleIndex] = RootsEligibility;
MerkleRootsAmounts[SaleIndex] = RootsAmounts;
}
/**
* @dev Changes The Sale Discounts
* @param SaleIndex The Sale Index To Change
* @param NewDiscountAmounts The New Discount Amounts To Change
*/
function ___ChangeDiscountAmounts(uint SaleIndex, uint[] calldata NewDiscountAmounts) external onlyAdmin { Discounts[SaleIndex] = NewDiscountAmounts; }
/**
* @dev Changes The MintPass ProjectID
* @param SaleIndex The Sale Index To Change
* @param MintPassProjectID The New MintPass ProjectID To Change
*/
function ___ChangeMintPassProjectID(uint SaleIndex, uint MintPassProjectID) external onlyAdmin { AuctionParams[SaleIndex]._ProjectIDMintPass = MintPassProjectID;}
/**
* @dev Changes The Auction Global Pause State At `SaleIndex`
* @param SaleIndex The Sale Index To Change
*/
function ___ChangeActiveState(uint SaleIndex) external onlyAdmin { SaleState[SaleIndex]._Active = !SaleState[SaleIndex]._Active; }
/**
* @dev Changes Min Bid
* @param SaleIndex The Sale Index To Change
* @param NewMinBid The New Minimum Bid
*/
function ___ChangeMinBid(uint SaleIndex, uint NewMinBid) external onlyAdmin { SaleState[SaleIndex]._LastMinBid = NewMinBid; }
/**
* @dev Changes If The Lowest Valid Leaderboard Bid Is What Everyone Pays
* @param SaleIndex The Sale Index To Change
* @param NewState The New State (Boolean) (True = Everyone Pays Lowest Leaderboard Bid, False = Everyone Pays Their Bid ETH Value)
*/
function ___ChangeSettlementEnabled(uint SaleIndex, bool NewState) external onlyAdmin { AuctionParams[SaleIndex]._SettlementEnabled = NewState; }
/**
* @dev Changes The Bid Threshold (Controls The Leaderboard Size)
* @param SaleIndex The Sale Index To Change
* @param NewLeaderboardSize The New Leaderboard Size
*/
function ___ChangeLeaderboardSize(uint SaleIndex, uint NewLeaderboardSize) external onlyAdmin { AuctionParams[SaleIndex]._LeaderboardSize = NewLeaderboardSize; }
/**
* @dev Changes The Unix Start Time
* @param SaleIndex The Sale Index To Change
* @param NewUnixStartTime The New Unix Start Time
*/
function ___ChangeUnixStartTime(uint SaleIndex, uint NewUnixStartTime) external onlyAdmin { AuctionParams[SaleIndex]._UnixStartTime = NewUnixStartTime; }
/**
* @dev Changes The Unix End Time
* @param SaleIndex The Sale Index To Change
* @param NewUnixEndTime The New Unix End Time
*/
function ___ChangeUnixEndTime(uint SaleIndex, uint NewUnixEndTime) external onlyAdmin { AuctionParams[SaleIndex]._UnixEndTime = NewUnixEndTime; }
/**
* @dev Changes The Minimum BIPs Increase
* @param SaleIndex The Sale Index To Change
* @param NewMinBIPSIncrease The New Minimum BIPs Increase (In BIPS) (105 = 5% Increase, 150 = 50% Increase etc...)
*/
function ___ChangeMinBIPSIncrease(uint SaleIndex, uint NewMinBIPSIncrease) external onlyAdmin { AuctionParams[SaleIndex]._MinBIPSIncrease = NewMinBIPSIncrease; }
/**
* @dev Changes The # Of Seconds The Auction Is Extended By If Auction End Time Is Within `AuctionParams[SaleIndex]._SecondsThreshold`
* @param SaleIndex The Sale Index To Change
* @param Seconds The New # Of Seconds To Extend Auction By
*/
function ___ChangeSecondsExtension(uint SaleIndex, uint Seconds) external onlyAdmin { AuctionParams[SaleIndex]._SecondsExtension = Seconds; }
/**
* @dev Changes The # Of Seconsd Within Auction End Time To Be Eligible For Auction Extension
* @param SaleIndex The Sale Index To Change
* @param Seconds The New # Of Seconds Within Auction End Time To Be Eligible For Auction Extension
*/
function ___ChangeSecondsThreshold(uint SaleIndex, uint Seconds) external onlyAdmin { AuctionParams[SaleIndex]._SecondsThreshold = Seconds; }
/**
* @dev Changes The Current NFT Address
* @param SaleIndex The Sale Index To Change
* @param NewAddress The New NFT Address
*/
function ___ChangeNFTAddress(uint SaleIndex, address NewAddress) external onlyAdmin { AuctionParams[SaleIndex]._NFT = NewAddress; }
/**
* @dev Changes The Current Operator Address (Address That Holds NFTs To Disperse)
* @param SaleIndex The Sale Index To Change
* @param Operator The New Operator Address (Address Holding NFTs To Disperse)
*/
function ___ChangeOperator(uint SaleIndex, address Operator) external onlyAdmin { AuctionParams[SaleIndex]._Operator = Operator; }
/**
* @dev Rebate All Unclaimed Bids & Sends Remaining ETH To Multisig
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function ___InitiateRebateAndProceeds(uint SaleIndex) external onlyAdmin
{
SaleState[SaleIndex]._Active = false; // Ends Auction
__AdminInitiateProceeds(SaleIndex); // Initiates Admin Withdraw Of Proceeds (MUST BE CALLED FIRST)
}
/**
* @dev Initiates Withdrawl Proceeds & Disperses NFTs To The Top Bidders On The Leaderboard (First-Come-First-Serve) (When TokenID Is Ambiguous)
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function ___ProcessETHAndNFTsTokenIDsAmbiguous(uint SaleIndex) external onlyAdmin
{
SaleState[SaleIndex]._Active = false; // Ends Auction
__AdminInitiateProceeds(SaleIndex); // Initiates Admin Withdraw Of Proceeds (MUST BE CALLED FIRST)
__DisperseNFTsByFCFSAmbiguous(SaleIndex); // Initiates Admin Disperse Of NFTs (MUST BE CALLED LAST)
}
/**
* @dev Initiates Withdrawl Proceeds & Disperses NFTs To The Top Bidders On The Leaderboard (Ascending Ranking) (When TokenID Matters)
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function ___ProcessETHAndNFTsTokenIDsDistinct(uint SaleIndex) external onlyAdmin
{
SaleState[SaleIndex]._Active = false; // Ends Auction
__AdminInitiateProceeds(SaleIndex); // Initiates Admin Withdraw Of Proceeds (MUST BE CALLED FIRST)
__DisperseNFTsByAscendingRank(SaleIndex); // Initiates Admin Disperse Of NFTs (MUST BE CALLED LAST)
}
/**
* @dev Initiates Withdrawl Proceeds & Disperses NFTs To The Top Bidders On The Leaderboard With Specific TokenIDs (No Ranking)
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function ___ProcessETHAndNFTsTokenIDsSpecificUnranked(uint SaleIndex, uint[] calldata TokenIDs) external onlyAdmin
{
SaleState[SaleIndex]._Active = false; // Ends Auction
__AdminInitiateProceeds(SaleIndex); // Initiates Admin Withdraw Of Proceeds (MUST BE CALLED FIRST)
__DisperseNFTsByUniqueTokenIDs(SaleIndex, TokenIDs); // Initiates Admin Disperse Of NFTs (MUST BE CALLED LAST)
}
/**
* @dev Initiates Withdrawl Proceeds & Disperses NFTs To The Top Bidders On The Leaderboard With Specific TokenIDs (Ascending Ranking)
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function ___ProcessETHAndNFTsTokenIDsSpecificRanked(uint SaleIndex, uint[] calldata TokenIDs) external onlyAdmin
{
SaleState[SaleIndex]._Active = false; // Ends Auction
__AdminInitiateProceeds(SaleIndex); // Initiates Admin Withdraw Of Proceeds (MUST BE CALLED FIRST)
__DisperseNFTsByUniqueTokenIDsAscending(SaleIndex, TokenIDs); // Initiates Admin Disperse Of NFTs (MUST BE CALLED LAST)
}
/**
* @dev Initiates Withdrawl Proceeds & Disperses NFTs To The Top Bidders On The Leaderboard With Mint Pack Amounts (Ascending Ranking)
*/
function ___ProcessETHAndNFTsMintPacksRanked(uint SaleIndex) external onlyAdmin
{
SaleState[SaleIndex]._Active = false; // Ends Auction
__AdminInitiateProceeds(SaleIndex); // Initiates Admin Withdraw Of Proceeds (MUST BE CALLED FIRST)
__DisperseNFTsMintPacks(SaleIndex); // Initiates Admin Disperse Of NFTs (MUST BE CALLED LAST)
}
/*------------------
* OWNER FUNCTIONS *
-------------------*/
/**
* @dev Adds An Admin
* @param Wallet The Wallet To Add As An Admin
*/
function ____AuthorizeAddress(address Wallet) external onlyOwner { Admin[Wallet] = true; }
/**
* @dev Removes An Admin
* @param Wallet The Wallet To Remove As An Admin
*/
function ____DeuthorizeAddress(address Wallet) external onlyOwner { Admin[Wallet] = false; }
/**
* @dev Initiates Withdrawl Proceeds For The Leaderboard
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function ____InitiateOnlyProceeds(uint SaleIndex) external onlyOwner { __AdminInitiateProceeds(SaleIndex); }
/**
* @dev Withdraws All Ether From The Contract
* @notice This Is A Safety Function To Prevent Ether Locking, Only Use In An Emergency
*/
function ____WithdrawEther() external onlyOwner { payable(msg.sender).transfer(address(this).balance); }
/**
* @dev Withdraws Ether From Contract To Address With An Amount
* @param Recipient The Recipient Of The Ether
* @param Amount The Amount Of Ether To Withdraw (In WEI)
* @notice This Is A Safety Function To Prevent Ether Locking, Only Use In An Emergency
*/
function ____WithdrawEtherToAddress(address payable Recipient, uint Amount) external onlyOwner
{
require(Amount > 0 && Amount <= address(this).balance, "Invalid Amount");
(bool Success, ) = Recipient.call{value: Amount}("");
require(Success, "Unable to Withdraw, Recipient May Have Reverted");
}
/*---------------------
* INTERNAL FUNCTIONS *
----------------------*/
/**
* @dev Calculates The Minimum Valid Bid And Seeds The Leaderboard
* @param SaleIndex The Sale Index To Calculate The Minimum Valid Bid For
* @param MsgValue The Message Value (In WEI) To Calculate The Minimum Valid Bid For
*/
function __FinalizeNewBid(uint SaleIndex, uint MsgValue) internal
{
(uint MinBid, uint LeaderboardIndex) = _ViewMinimumValidBidAndIndex(SaleIndex);
require(MsgValue >= MinBid, "Auction: Bid Amount Must Be >= Current Leaderboard Floor * 1.05"); // Requires Min Bid
bool Valid = (SaleState[SaleIndex]._GlobalUniqueBids >= AuctionParams[SaleIndex]._LeaderboardSize);
if(Valid)
{
__KickbackETH(SaleIndex, LeaderboardIndex); // Rebate ETH To Previous Bidder
SaleState[SaleIndex]._LastMinBid = MinBid; // Stores Minimum Bid
}
Leaderboard[SaleIndex][LeaderboardIndex] = SaleState[SaleIndex]._GlobalUniqueBids; // Kicks Old Bid Index Out Of Leaderboard
}
/**
* @dev Kicks Losing Bidder's ETH Back To Them
* @param SaleIndex The Sale Index To Trigger Disbursement
* @param LeaderboardIndex The Leaderboard Index To Kickback ETH To
*/
function __KickbackETH(uint SaleIndex, uint LeaderboardIndex) internal
{
if(!Bids[SaleIndex][Leaderboard[SaleIndex][LeaderboardIndex]]._Rebated)
{
Bids[SaleIndex][Leaderboard[SaleIndex][LeaderboardIndex]]._Rebated = true;
(bool Success,) = Bids[SaleIndex][Leaderboard[SaleIndex][LeaderboardIndex]]._Bidder.call
{ value: Bids[SaleIndex][Leaderboard[SaleIndex][LeaderboardIndex]]._ETHValue }("");
require(Success, "Auction: Kickback Failed");
emit BidRefunded(Bids[SaleIndex][Leaderboard[SaleIndex][LeaderboardIndex]]._OGBidIndex);
}
}
/**
* @dev Finalizes ETH From User's Pending Bid(s)
* @param SaleIndex The Sale Index To Trigger Disbursement
* @param Bidder The Bidder To Disperse ETH To
*/
function __UserDisperseETH(uint SaleIndex, address Bidder) internal
{
uint[] memory _UserBidIndexes = UserBidIndexes[SaleIndex][Bidder];
uint LLB = _ViewLowestLeaderboardBid(SaleIndex);
uint TotalRebate;
uint TotalPaid;
uint CurrentRebate;
uint CurrentPaid;
uint Discount;
for(uint x; x < UserBidIndexes[SaleIndex][Bidder].length; x++)
{
if (
!Bids[SaleIndex][_UserBidIndexes[x]]._Winner // Bid Is Not Winner
&&
!Bids[SaleIndex][_UserBidIndexes[x]]._Rebated // Bid Has Not Been Rebated
&&
Bids[SaleIndex][_UserBidIndexes[x]]._ETHValue >= LLB // Bid Is Valid ETH Value
)
{
Bids[SaleIndex][UserBidIndexes[SaleIndex][Bidder][x]]._Winner = true;
if(Bids[SaleIndex][_UserBidIndexes[x]]._Priority != 69420) { Discount = Discounts[SaleIndex][Bids[SaleIndex][_UserBidIndexes[x]]._Priority]; }
else { Discount = 100; }
if(AuctionParams[SaleIndex]._SettlementEnabled)
{
Bids[SaleIndex][_UserBidIndexes[x]]._Rebated = true;
CurrentRebate = ((Bids[SaleIndex][_UserBidIndexes[x]]._ETHValue - LLB) * Discount) / 100;
CurrentPaid = Bids[SaleIndex][_UserBidIndexes[x]]._ETHValue - CurrentRebate;
TotalRebate += CurrentRebate;
}
else { CurrentPaid = Bids[SaleIndex][_UserBidIndexes[x]]._ETHValue; }
TotalPaid += (CurrentPaid * Discount) / 100;
}
}
(bool MultisigWithdraw, ) = _BRT_MULTISIG.call { value: TotalPaid }("");
require(MultisigWithdraw, "Auction: Multisig Withdraw Failed");
if(TotalRebate > 0)
{
(bool UserWithdraw, ) = Bidder.call { value: TotalRebate }("");
require(UserWithdraw, "Auction: User Withdraw Failed");
}
}
/**
* @dev Disperses NFTs
* @param SaleIndex The Sale Index To Trigger Disbursement
* @param Bidder The Bidder To Disperse NFTs To
*/
function __UserDisperseNFT(uint SaleIndex, address Bidder) internal
{
IERC721 _NFT = IERC721(AuctionParams[SaleIndex]._NFT);
for(uint x; x < UserBidIndexes[SaleIndex][Bidder].length; x++)
{
if(
Bids[SaleIndex][UserBidIndexes[SaleIndex][Bidder][x]]._Winner
&&
!Bids[SaleIndex][UserBidIndexes[SaleIndex][Bidder][x]]._NFTSent
)
{
Bids[SaleIndex][UserBidIndexes[SaleIndex][Bidder][x]]._NFTSent = true;
_NFT._MintToFactory(Bidder, 1);
}
}
}
/**
* @dev Initiates Proceeds From The Leaderboard
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function __AdminInitiateProceeds(uint SaleIndex) internal
{
uint TotalProceeds;
uint LLB = _ViewLowestLeaderboardBid(SaleIndex);
uint RebateAmount;
uint Discount;
uint Priority;
uint ETHValue;
for(uint x; x < AuctionParams[SaleIndex]._LeaderboardSize; x++)
{
if (
!Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Rebated
&&
!Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Winner
&&
Bids[SaleIndex][Leaderboard[SaleIndex][x]]._ETHValue >= LLB
) {
Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Winner = true;
Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Rebated = true;
Priority = Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Priority;
ETHValue = Bids[SaleIndex][Leaderboard[SaleIndex][x]]._ETHValue;
if(Priority == 69420) { Discount = 100; } // No Merkle Priority
else { Discount = Discounts[SaleIndex][Priority]; } // Merkle Priority
if(AuctionParams[SaleIndex]._SettlementEnabled) { RebateAmount = ETHValue - ((LLB * Discount) / 100); }
else { RebateAmount = ETHValue - ((ETHValue * Discount) / 100); }
if(RebateAmount > 0)
{
(bool Rebate, ) = Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Bidder.call { value: RebateAmount }("");
require(Rebate, "Auction: Failed To Rebate ETH To Bidder, Use Failsafe Withdraw");
}
TotalProceeds += (ETHValue - RebateAmount);
}
}
(bool MultisigWithdraw, ) = _BRT_MULTISIG.call { value: TotalProceeds }("");
require(MultisigWithdraw, "Auction: Admin Failed To Withdraw ETH To Multisig, Use Failsafe Withdraw");
}
/**
* @dev Validates The Auction End Time & Extends If Necessary
* @param SaleIndex The Sale Index To Check
*/
function __CheckAndSeedAuctionEndTime(uint SaleIndex) internal
{
// Extends Auction If Rebate Last Price (Settlement Price) Is Not Enabled (For Sales Where Leaderboard Placement Matters)
if((AuctionParams[SaleIndex]._UnixEndTime - block.timestamp) < AuctionParams[SaleIndex]._SecondsThreshold) // If Bid Placed In Last 5 Minutes
{
AuctionParams[SaleIndex]._UnixEndTime = block.timestamp + AuctionParams[SaleIndex]._SecondsExtension; // Extends Auction By The # Of Configured Seconds
emit AuctionExtended();
}
}
/**
* @dev Disperses NFTs With Unique TokenIDs
* @param TokenIDs Array Of TokenIDs To Be Dispersed
*/
function __DisperseNFTsByUniqueTokenIDs(uint SaleIndex, uint[] calldata TokenIDs) internal
{
require(!SaleState[SaleIndex]._Active, "Auction: Auction Is Still Active, Must Disperse Funds & Finalize Auction First");
require(!SaleState[SaleIndex]._NFTsDispersed, "Auction: NFTs Already Dispersed");
require(TokenIDs.length == AuctionParams[SaleIndex]._LeaderboardSize, "Auction: TokenIDs Array Length Must Match Leaderboard Size");
address _Op = AuctionParams[SaleIndex]._Operator;
IERC721 _NFT = IERC721(AuctionParams[SaleIndex]._NFT);
for(uint x; x < TokenIDs.length; x++)
{
require(!NFTTokenIDHasBeenSent[SaleIndex][TokenIDs[x]], "Auction: TokenID Already Sent");
NFTTokenIDHasBeenSent[SaleIndex][TokenIDs[x]] = true;
_NFT.transferFrom(_Op, Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Bidder, TokenIDs[x]);
}
}
/**
* @dev Disperses NFTs With Unique TokenIDs
* @param TokenIDs Array Of TokenIDs To Be Dispersed
*/
function __DisperseNFTsByUniqueTokenIDsAscending(uint SaleIndex, uint[] calldata TokenIDs) internal
{
require(!SaleState[SaleIndex]._Active, "Auction: Auction Is Still Active, Must Disperse Funds & Finalize Auction First");
require(!SaleState[SaleIndex]._NFTsDispersed, "Auction: NFTs Already Dispersed");
require(TokenIDs.length == AuctionParams[SaleIndex]._LeaderboardSize, "Auction: TokenIDs Array Length Must Match Leaderboard Size");
uint[] memory _Ind = _ViewSortedLeaderboardBidIndexes(SaleIndex);
address _Op = AuctionParams[SaleIndex]._Operator;
IERC721 _NFT = IERC721(AuctionParams[SaleIndex]._NFT);
for(uint x; x < TokenIDs.length; x++)
{
require(!NFTTokenIDHasBeenSent[SaleIndex][TokenIDs[x]], "Auction: TokenID Already Sent");
NFTTokenIDHasBeenSent[SaleIndex][TokenIDs[x]] = true;
_NFT.transferFrom(_Op, Bids[SaleIndex][_Ind[x]]._Bidder, TokenIDs[x]);
}
}
/**
* @dev Disperses NFTs To The Top Bidders On The Leaderboard (First-Come-First-Serve) (Use When TokenID Is Ambiguous)
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function __DisperseNFTsByFCFSAmbiguous(uint SaleIndex) internal
{
require(!SaleState[SaleIndex]._Active, "Auction: Auction Is Still Active, Must Disperse Funds & Finalize Auction First");
require(!SaleState[SaleIndex]._NFTsDispersed, "Auction: NFTs Already Dispersed");
SaleState[SaleIndex]._NFTsDispersed = true;
IERC721 _NFT = IERC721(AuctionParams[SaleIndex]._NFT);
for(uint x; x < AuctionParams[SaleIndex]._LeaderboardSize; x++)
{
if(Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Winner && !Bids[SaleIndex][Leaderboard[SaleIndex][x]]._NFTSent)
{
Bids[SaleIndex][Leaderboard[SaleIndex][x]]._NFTSent = true;
_NFT._MintToFactory(Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Bidder, 1);
}
}
}
/**
* @dev Disperses NFTs By Ascending Ranking Of The Leaderboard (Use When TokenID Matters)
* @param SaleIndex The Sale Index To Trigger Disbursement
*/
function __DisperseNFTsByAscendingRank(uint SaleIndex) internal
{
require(SaleState[SaleIndex]._Active == false, "Auction: Auction Is Still Active, Must Disperse Funds First");
require(!SaleState[SaleIndex]._NFTsDispersed, "Auction: NFTs Already Dispersed");
SaleState[SaleIndex]._NFTsDispersed = true;
IERC721 _NFT = IERC721(AuctionParams[SaleIndex]._NFT);
uint[] memory _Ind = _ViewSortedLeaderboardBidIndexes(SaleIndex);
for(uint x; x < _Ind.length; x++)
{
if(Bids[SaleIndex][_Ind[x]]._Winner && !Bids[SaleIndex][_Ind[x]]._NFTSent)
{
Bids[SaleIndex][_Ind[x]]._NFTSent = true;
_NFT._MintToFactory(Bids[SaleIndex][_Ind[x]]._Bidder, 1);
}
}
}
/**
* @dev Disperses NFTs To The Top Bidders On The Leaderboard (With Mint Pack Amounts) (With Ranking Enabled)
*/
function __DisperseNFTsMintPacks(uint SaleIndex) internal
{
require(!SaleState[SaleIndex]._Active, "Auction: Auction Is Still Active, Must Disperse Funds & Finalize Auction First");
require(!SaleState[SaleIndex]._NFTsDispersed, "Auction: NFTs Already Dispersed");
require(AuctionParams[SaleIndex]._MintPackAmount > 1, "Auction: Mint Pack Amount Must Be > 1");
SaleState[SaleIndex]._NFTsDispersed = true;
IERC721 _NFT = IERC721(AuctionParams[SaleIndex]._NFT);
uint[] memory _Ind = _ViewSortedLeaderboardBidIndexes(SaleIndex);
uint _MintPackAmount = AuctionParams[SaleIndex]._MintPackAmount;
for(uint x; x < _Ind.length; x++)
{
if(Bids[SaleIndex][_Ind[x]]._Winner && !Bids[SaleIndex][_Ind[x]]._NFTSent)
{
Bids[SaleIndex][_Ind[x]]._NFTSent = true;
_NFT._MintToFactory(Bids[SaleIndex][_Ind[x]]._Bidder, _MintPackAmount);
}
}
}
/*------------------------
* PUBLIC VIEW FUNCTIONS *
-------------------------*/
/**
* @dev Returns All Necessary Leaderboard Components
* @param SaleIndex The Sale Index To View
* @param Wallet The Wallet Address Of The Bidder ('0x0000000000000000000000000000000000000000') If No Wallet
*/
function ViewFrontend ( uint SaleIndex, address Wallet ) public view returns (
uint _LLB,
uint _MVB,
AllAuctionParams memory _AuctionParams,
Bid[] memory _RankedLeaderboard,
Bid[] memory _MasterBids,
uint[] memory _UserBidIndexes
) {
uint LLB = _ViewLowestLeaderboardBid(SaleIndex);
_MasterBids = _ViewBidsUnique(SaleIndex);
_RankedLeaderboard = _ViewLeaderboardRanked(SaleIndex);
uint MVB = LLB * (AuctionParams[SaleIndex]._MinBIPSIncrease) / 100;
if(SaleState[SaleIndex]._GlobalUniqueBids < AuctionParams[SaleIndex]._LeaderboardSize)
{ (LLB, MVB) = (SaleState[SaleIndex]._LastMinBid, SaleState[SaleIndex]._LastMinBid); }
return (
LLB,
MVB,
ViewAuctionParams(SaleIndex),
ViewLeaderboardRanked(SaleIndex),
_MasterBids,
UserBidIndexes[SaleIndex][Wallet]
);
}
/**
* @dev Returns All Bid Values In The Leaderboard
* @param SaleIndex The Sale Index To View
*/
function ViewLeaderboardBids(uint SaleIndex) public view returns (uint[] memory)
{
uint[] memory _Indexes = ViewLeaderboardIndexes(SaleIndex);
uint[] memory _BidValues = new uint[](AuctionParams[SaleIndex]._LeaderboardSize);
for(uint x; x < AuctionParams[SaleIndex]._LeaderboardSize; x++) { _BidValues[x] = Bids[SaleIndex][_Indexes[x]]._ETHValue; }
return _BidValues;
}
/**
* @dev Returns A Bid Array Of Ranked Top Bids
* @param SaleIndex The Sale Index To View
*/
function ViewLeaderboardRanked(uint SaleIndex) public view returns(Bid[] memory) { return _ViewLeaderboardRanked(SaleIndex); }
/**
* @dev Returns All Bid Indexes In The Leaderboard
* @param SaleIndex The Sale Index To View
*/
function ViewLeaderboardIndexes(uint SaleIndex) public view returns (uint[] memory)
{
uint[] memory _LeaderboardIndexes = new uint[](AuctionParams[SaleIndex]._LeaderboardSize);
for(uint x; x < AuctionParams[SaleIndex]._LeaderboardSize; x++) { _LeaderboardIndexes[x] = Leaderboard[SaleIndex][x]; }
return _LeaderboardIndexes;
}
/**
* @dev Returns The Minimum Valid Bid
* @param SaleIndex The Sale Index To View
*/
function ViewMinimumValidBid(uint SaleIndex) public view returns (uint ValidBid)
{
(ValidBid,) = _ViewMinimumValidBidAndIndex(SaleIndex);
return ValidBid;
}
/**
* @dev Returns A 'Bid' Struct Array Corresponding To Input Indexes
* @param SaleIndex The Sale Index To View
*/
function ViewBidsUnique(uint SaleIndex) public view returns ( Bid[] memory )
{
Bid[] memory _Bids = new Bid[](SaleState[SaleIndex]._GlobalUniqueBids);
for(uint x; x < SaleState[SaleIndex]._GlobalUniqueBids; x++) { _Bids[x] = Bids[SaleIndex][x]; }
return _Bids;
}
/**
* @dev Returns A 'Bid' Struct Array Corresponding To Input Indexes
* @param SaleIndex The Sale Index To View
* @param Indexes The Indexes To Return
*/
function ViewBidsAtIndexes(uint SaleIndex, uint[] calldata Indexes) public view returns(Bid[] memory)
{
Bid[] memory _Bids = new Bid[](Indexes.length);
for(uint x; x < Indexes.length; x++) { _Bids[x] = Bids[SaleIndex][Indexes[x]]; }
return _Bids;
}
/**
* @dev Returns A `Bid` Struct Array Of All Unique Bids In The Auction Submitted By `Wallet`
* @param SaleIndex The Sale Index To View
* @param Wallet The Wallet Address To View
*/
function ViewWalletBids(uint SaleIndex, address Wallet) public view returns(Bid[] memory)
{
uint[] memory _Indexes = UserBidIndexes[SaleIndex][Wallet];
Bid[] memory _Bids = new Bid[](_Indexes.length);
for(uint x; x < _Indexes.length; x++) { _Bids[x] = Bids[SaleIndex][_Indexes[x]]; }
return _Bids;
}
/**
* @dev Returns An Array Of `Wallet` Submitted Bid Indexes
* @param SaleIndex The Sale Index To View
* @param Wallet The Wallet Address To View
*/
function ViewWalletBidIndexes(uint SaleIndex, address Wallet) public view returns(uint[] memory) { return UserBidIndexes[SaleIndex][Wallet]; }
/**
* @dev Returns All Of The Current Auction Parameters
* @param SaleIndex The Sale Index To View
*/
function ViewAuctionParams(uint SaleIndex) public view returns (AllAuctionParams memory)
{
return AllAuctionParams (
AuctionParams[SaleIndex]._Name,
SaleState[SaleIndex]._Active,
SaleState[SaleIndex]._NFTsDispersed,
AuctionParams[SaleIndex]._SettlementEnabled,
AuctionParams[SaleIndex]._LeaderboardSize,
AuctionParams[SaleIndex]._UnixStartTime,
AuctionParams[SaleIndex]._UnixEndTime,
AuctionParams[SaleIndex]._MinBIPSIncrease,
AuctionParams[SaleIndex]._SecondsExtension,
AuctionParams[SaleIndex]._SecondsThreshold,
SaleState[SaleIndex]._LastMinBid,
SaleState[SaleIndex]._GlobalUniqueBids,
AuctionParams[SaleIndex]._ProjectIDMintPass,
AuctionParams[SaleIndex]._MintPackAmount,
AuctionParams[SaleIndex]._NFT
);
}
/**
* @dev Returns Merkle Roots For A Specific Sale
* @param SaleIndex The Sale Index To View
*/
function ViewRoots(uint SaleIndex) public view returns (bytes32[] memory) { return MerkleRootsEligibility[SaleIndex]; }
/*--------------------------
* INTERNAL VIEW FUNCTIONS *
---------------------------*/
/**
* @dev Validates Merkle Proof And Returns Merkle Priority
* @param SaleIndex The Sale Index To View
* @param MaxAmount The Maximum Allocation For Merkle Priority
* @param Bidder The Bidder To Validate
* @param ProofEligibility The Merkle Proof To Validate For Priority Eligibilty Tier
* @param ProofAmount The Merkle Proof To Validate For Maximum Merkle Amount
*/
function _ValidateMerkleProofs (
uint SaleIndex,
uint MaxAmount,
address Bidder,
bytes32[] calldata ProofEligibility,
bytes32[] calldata ProofAmount
) internal view returns (uint) {
bytes32 Leaf = keccak256(abi.encodePacked(Bidder));
for(uint Priority; Priority < MerkleRootsEligibility[SaleIndex].length; Priority++)
{
if(MerkleProof.verify(ProofEligibility, MerkleRootsEligibility[SaleIndex][Priority], Leaf))
{
require (
MerkleProof.verify (
ProofAmount,
MerkleRootsAmounts[SaleIndex][Priority],
keccak256(abi.encodePacked(Bidder, MaxAmount))
),
"Auction: Invalid Merkle Amount"
);
return ( Priority ); // Returns Valid Priority
}
}
return ( 69420 ); // Returns Default Out Of Bounds Priority
}
/**
* @dev Returns The Leaderboard Index Of The Smallest Bid In The Leaderboard
* @param SaleIndex The Sale Index To View
*/
function _ViewMinimumValidLeaderboardIndex(uint SaleIndex) internal view returns (uint)
{
uint CurrentMinBid = type(uint).max;
uint LeaderboardIndexToReplace;
uint ETHValue;
if(SaleState[SaleIndex]._GlobalUniqueBids < AuctionParams[SaleIndex]._LeaderboardSize) { return SaleState[SaleIndex]._GlobalUniqueBids; }
for(uint IndexLeaderboard; IndexLeaderboard < AuctionParams[SaleIndex]._LeaderboardSize; IndexLeaderboard++)
{
ETHValue = Bids[SaleIndex][Leaderboard[SaleIndex][IndexLeaderboard]]._ETHValue;
if(ETHValue <= CurrentMinBid)
{
CurrentMinBid = ETHValue;
LeaderboardIndexToReplace = IndexLeaderboard;
}
}
return LeaderboardIndexToReplace;
}
/**
* @dev Returns The Lowest Bid In The Leaderboard
* @param SaleIndex The Sale Index To View
*/
function _ViewLowestLeaderboardBid(uint SaleIndex) internal view returns (uint LLB)
{
LLB = type(uint).max;
for(uint x; x < AuctionParams[SaleIndex]._LeaderboardSize; x++)
{
if(Bids[SaleIndex][Leaderboard[SaleIndex][x]]._ETHValue < LLB) { LLB = Bids[SaleIndex][Leaderboard[SaleIndex][x]]._ETHValue; }
}
return LLB;
}
/**
* @dev Returns The Minimum Valid Bid Which Is The Current Lowest Bid In The Leaderboard * 1.05
* @param SaleIndex The Sale Index To View
*/
function _ViewMinimumValidBidAndIndex(uint SaleIndex) internal view returns (uint, uint)
{
uint LeaderboardIndex = _ViewMinimumValidLeaderboardIndex(SaleIndex);
return (
SaleState[SaleIndex]._GlobalUniqueBids < AuctionParams[SaleIndex]._LeaderboardSize
? // If Unique Bids Less Than LeaderboardSize Return NewMinimumBid & Eligible LeaderboardIndex
(SaleState[SaleIndex]._LastMinBid, LeaderboardIndex)
: // Else Return NewMinimumBid & Eligible LeaderboardIndex
((Bids[SaleIndex][Leaderboard[SaleIndex][LeaderboardIndex]]._ETHValue * AuctionParams[SaleIndex]._MinBIPSIncrease) / 100, LeaderboardIndex)
);
}
/**
* @dev Returns A 'Bid' Struct Array Of All Unique Bids In The Auction
* @param SaleIndex The Sale Index To View
* note: this will throw `out of gas` after 1648~ unique bids because block gas limit is 30M, use `ViewBids()` with indexes after 1648~ unique bids
*/
function _ViewBidsUnique(uint SaleIndex) internal view returns(Bid[] memory)
{
uint GlobalUniqueBids = SaleState[SaleIndex]._GlobalUniqueBids;
Bid[] memory _Bids = new Bid[](GlobalUniqueBids);
for(uint x; x < GlobalUniqueBids; x++) { _Bids[x] = Bids[SaleIndex][x]; }
return _Bids;
}
/**
* @dev Returns A Bid Array Of Ranked Top Bids
* @param SaleIndex The Sale Index To View
*/
function _ViewLeaderboardRanked(uint SaleIndex) internal view returns(Bid[] memory)
{
uint[] memory _Ind = _ViewSortedLeaderboardBidIndexes(SaleIndex);
Bid[] memory _Leaderboard = new Bid[](_Ind.length);
for(uint x; x < _Ind.length; x++) { _Leaderboard[x] = Bids[SaleIndex][_Ind[x]]; }
return _Leaderboard;
}
/**
* @dev Returns A Sorted List Of ETH Bids @ '[n][0]' & The Indexes Of The Original Bids @ '[n][1]' & The Timestamps @ '[n][2]'
* @param SaleIndex The Sale Index To View
* note: This Will Give Priority To Earlier Bid Indexes & Timestamps
* note: insertion sort O(n^2) seemed like best approach because english auction bids increase as auction progresses, otherwise quicksort prob better O(nlogn)
* note: because block gas limit is 30M, this will `out-of-gas` dependant on how much sorting needs done if you have a more eloquent way of doing this hmu
* note: you should (in general) not sort large things in solidity (as of 0.8~) because it is very gas inefficient, this is just for demonstration purposes
*/
function _ViewSortedLeaderboardBidIndexes(uint SaleIndex) internal view returns (uint[] memory)
{
uint Size;
if(SaleState[SaleIndex]._GlobalUniqueBids < AuctionParams[SaleIndex]._LeaderboardSize) { Size = SaleState[SaleIndex]._GlobalUniqueBids; }
else { Size = AuctionParams[SaleIndex]._LeaderboardSize; }
uint[][] memory BidsAndIndexes = new uint[][](Size);
for(uint x; x < BidsAndIndexes.length; x++)
{
BidsAndIndexes[x] = new uint[](3); // Init Sub-Array
BidsAndIndexes[x][0] = Bids[SaleIndex][Leaderboard[SaleIndex][x]]._ETHValue; // Assign [x][0] -> ETHValue
BidsAndIndexes[x][1] = Leaderboard[SaleIndex][x]; // Assign [x][1] -> Original Index
BidsAndIndexes[x][2] = Bids[SaleIndex][Leaderboard[SaleIndex][x]]._Timestamp; // Assign [x][2] -> Timestamp
}
for(uint i; i < BidsAndIndexes.length; i++)
{
uint ETHValue = BidsAndIndexes[i][0]; // Preserve ETHValue
uint OGBidIndex = BidsAndIndexes[i][1]; // Preserve OGBidIndex
uint Timestamp = BidsAndIndexes[i][2]; // Preserve Timestamp
uint j = i;
while(j > 0 && BidsAndIndexes[j-1][0] >= ETHValue)
{
if(
BidsAndIndexes[j-1][0] == ETHValue && BidsAndIndexes[j-1][1] > OGBidIndex // Preserve Lower Original Index
||
BidsAndIndexes[j-1][0] == ETHValue && BidsAndIndexes[j-1][2] > Timestamp // Preserve Lower Timestamp
) { break; }
BidsAndIndexes[j][0] = BidsAndIndexes[j-1][0]; // Move Larger Element To The Right
BidsAndIndexes[j][1] = BidsAndIndexes[j-1][1]; // Move OG Index
BidsAndIndexes[j][2] = BidsAndIndexes[j-1][2]; // Move Timestamp
j--;
}
BidsAndIndexes[j][0] = ETHValue; // Insert ETHValue In Correct Location
BidsAndIndexes[j][1] = OGBidIndex; // Insert OGBidIndex In Correct Location
BidsAndIndexes[j][2] = Timestamp; // Insert Timestamp In Correct Location
}
uint[] memory SortedBidIndexes = new uint[](Size);
for(uint y; y < BidsAndIndexes.length; y++) { SortedBidIndexes[Size - 1 - y] = BidsAndIndexes[y][1]; }
return SortedBidIndexes;
}
/**
* @dev onlyAdmin Modifier
*/
modifier onlyAdmin
{
require(Admin[msg.sender], "Auction: onlyAdmin: Caller Is Not Admin");
_;
}
}
/**
* @dev Interface For ERC721 Contracts
*/
interface IERC721
{
/**
* @dev Mints A NFT From Custom Smart Contract Directly
*/
function _MintToFactory(address Recipient, uint Amount) external;
/**
* @dev Transfers An Already Minted NFT
*/
function transferFrom(address from, address to, uint tokenID) external;
}
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)
pragma solidity ^0.8.0;
/**
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
//SPDX-License-Identifier: MIT
/**
* @title DelegateCashEnabled
* @author @brougkr
* @notice For Easily Integrating `delegate.cash`
*/
pragma solidity 0.8.19;
abstract contract DelegateCashEnabled
{
address private constant _DN = 0x00000000000076A84feF008CDAbe6409d2FE638B;
IDelegation public constant DelegateCash = IDelegation(_DN);
}
interface IDelegation
{
/**
* @dev Returns If A Vault Has Delegated To The Delegate
*/
function checkDelegateForAll(address delegate, address vault) external view returns (bool);
}
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (utils/cryptography/MerkleProof.sol)
pragma solidity ^0.8.0;
/**
* @dev These functions deal with verification of Merkle Tree proofs.
*
* The tree and the proofs can be generated using our
* https://github.com/OpenZeppelin/merkle-tree[JavaScript library].
* You will find a quickstart guide in the readme.
*
* WARNING: You should avoid using leaf values that are 64 bytes long prior to
* hashing, or use a hash function other than keccak256 for hashing leaves.
* This is because the concatenation of a sorted pair of internal nodes in
* the merkle tree could be reinterpreted as a leaf value.
* OpenZeppelin's JavaScript library generates merkle trees that are safe
* against this attack out of the box.
*/
library MerkleProof {
/**
* @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree
* defined by `root`. For this, a `proof` must be provided, containing
* sibling hashes on the branch from the leaf to the root of the tree. Each
* pair of leaves and each pair of pre-images are assumed to be sorted.
*/
function verify(
bytes32[] memory proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProof(proof, leaf) == root;
}
/**
* @dev Calldata version of {verify}
*
* _Available since v4.7._
*/
function verifyCalldata(
bytes32[] calldata proof,
bytes32 root,
bytes32 leaf
) internal pure returns (bool) {
return processProofCalldata(proof, leaf) == root;
}
/**
* @dev Returns the rebuilt hash obtained by traversing a Merkle tree up
* from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt
* hash matches the root of the tree. When processing the proof, the pairs
* of leafs & pre-images are assumed to be sorted.
*
* _Available since v4.4._
*/
function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
computedHash = _hashPair(computedHash, proof[i]);
}
return computedHash;
}
/**
* @dev Calldata version of {processProof}
*
* _Available since v4.7._
*/
function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) {
bytes32 computedHash = leaf;
for (uint256 i = 0; i < proof.length; i++) {
computedHash = _hashPair(computedHash, proof[i]);
}
return computedHash;
}
/**
* @dev Returns true if the `leaves` can be simultaneously proven to be a part of a merkle tree defined by
* `root`, according to `proof` and `proofFlags` as described in {processMultiProof}.
*
* CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
*
* _Available since v4.7._
*/
function multiProofVerify(
bytes32[] memory proof,
bool[] memory proofFlags,
bytes32 root,
bytes32[] memory leaves
) internal pure returns (bool) {
return processMultiProof(proof, proofFlags, leaves) == root;
}
/**
* @dev Calldata version of {multiProofVerify}
*
* CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
*
* _Available since v4.7._
*/
function multiProofVerifyCalldata(
bytes32[] calldata proof,
bool[] calldata proofFlags,
bytes32 root,
bytes32[] memory leaves
) internal pure returns (bool) {
return processMultiProofCalldata(proof, proofFlags, leaves) == root;
}
/**
* @dev Returns the root of a tree reconstructed from `leaves` and sibling nodes in `proof`. The reconstruction
* proceeds by incrementally reconstructing all inner nodes by combining a leaf/inner node with either another
* leaf/inner node or a proof sibling node, depending on whether each `proofFlags` item is true or false
* respectively.
*
* CAUTION: Not all merkle trees admit multiproofs. To use multiproofs, it is sufficient to ensure that: 1) the tree
* is complete (but not necessarily perfect), 2) the leaves to be proven are in the opposite order they are in the
* tree (i.e., as seen from right to left starting at the deepest layer and continuing at the next layer).
*
* _Available since v4.7._
*/
function processMultiProof(
bytes32[] memory proof,
bool[] memory proofFlags,
bytes32[] memory leaves
) internal pure returns (bytes32 merkleRoot) {
// This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by
// consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
// `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
// the merkle tree.
uint256 leavesLen = leaves.length;
uint256 totalHashes = proofFlags.length;
// Check proof validity.
require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");
// The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
// `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
bytes32[] memory hashes = new bytes32[](totalHashes);
uint256 leafPos = 0;
uint256 hashPos = 0;
uint256 proofPos = 0;
// At each step, we compute the next hash using two values:
// - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
// get the next hash.
// - depending on the flag, either another value for the "main queue" (merging branches) or an element from the
// `proof` array.
for (uint256 i = 0; i < totalHashes; i++) {
bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
hashes[i] = _hashPair(a, b);
}
if (totalHashes > 0) {
return hashes[totalHashes - 1];
} else if (leavesLen > 0) {
return leaves[0];
} else {
return proof[0];
}
}
/**
* @dev Calldata version of {processMultiProof}.
*
* CAUTION: Not all merkle trees admit multiproofs. See {processMultiProof} for details.
*
* _Available since v4.7._
*/
function processMultiProofCalldata(
bytes32[] calldata proof,
bool[] calldata proofFlags,
bytes32[] memory leaves
) internal pure returns (bytes32 merkleRoot) {
// This function rebuild the root hash by traversing the tree up from the leaves. The root is rebuilt by
// consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the
// `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of
// the merkle tree.
uint256 leavesLen = leaves.length;
uint256 totalHashes = proofFlags.length;
// Check proof validity.
require(leavesLen + proof.length - 1 == totalHashes, "MerkleProof: invalid multiproof");
// The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using
// `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop".
bytes32[] memory hashes = new bytes32[](totalHashes);
uint256 leafPos = 0;
uint256 hashPos = 0;
uint256 proofPos = 0;
// At each step, we compute the next hash using two values:
// - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we
// get the next hash.
// - depending on the flag, either another value for the "main queue" (merging branches) or an element from the
// `proof` array.
for (uint256 i = 0; i < totalHashes; i++) {
bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++];
bytes32 b = proofFlags[i] ? leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++] : proof[proofPos++];
hashes[i] = _hashPair(a, b);
}
if (totalHashes > 0) {
return hashes[totalHashes - 1];
} else if (leavesLen > 0) {
return leaves[0];
} else {
return proof[0];
}
}
function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) {
return a < b ? _efficientHash(a, b) : _efficientHash(b, a);
}
function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) {
/// @solidity memory-safe-assembly
assembly {
mstore(0x00, a)
mstore(0x20, b)
value := keccak256(0x00, 0x40)
}
}
}
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)
pragma solidity ^0.8.0;
import "../utils/Context.sol";
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership}.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor() {
_transferOwnership(_msgSender());
}
/**
* @dev Throws if called by any account other than the owner.
*/
modifier onlyOwner() {
_checkOwner();
_;
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view virtual returns (address) {
return _owner;
}
/**
* @dev Throws if the sender is not the owner.
*/
function _checkOwner() internal view virtual {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore. Can only be called by the current owner.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*/
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Internal function without access restriction.
*/
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (security/ReentrancyGuard.sol)
pragma solidity ^0.8.0;
/**
* @dev Contract module that helps prevent reentrant calls to a function.
*
* Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
* available, which can be applied to functions to make sure there are no nested
* (reentrant) calls to them.
*
* Note that because there is a single `nonReentrant` guard, functions marked as
* `nonReentrant` may not call one another. This can be worked around by making
* those functions `private`, and then adding `external` `nonReentrant` entry
* points to them.
*
* TIP: If you would like to learn more about reentrancy and alternative ways
* to protect against it, check out our blog post
* https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
*/
abstract contract ReentrancyGuard {
// Booleans are more expensive than uint256 or any type that takes up a full
// word because each write operation emits an extra SLOAD to first read the
// slot's contents, replace the bits taken up by the boolean, and then write
// back. This is the compiler's defense against contract upgrades and
// pointer aliasing, and it cannot be disabled.
// The values being non-zero value makes deployment a bit more expensive,
// but in exchange the refund on every call to nonReentrant will be lower in
// amount. Since refunds are capped to a percentage of the total
// transaction's gas, it is best to keep them low in cases like this one, to
// increase the likelihood of the full refund coming into effect.
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
/**
* @dev Prevents a contract from calling itself, directly or indirectly.
* Calling a `nonReentrant` function from another `nonReentrant`
* function is not supported. It is possible to prevent this from happening
* by making the `nonReentrant` function external, and making it call a
* `private` function that does the actual work.
*/
modifier nonReentrant() {
_nonReentrantBefore();
_;
_nonReentrantAfter();
}
function _nonReentrantBefore() private {
// On the first call to nonReentrant, _status will be _NOT_ENTERED
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
// Any calls to nonReentrant after this point will fail
_status = _ENTERED;
}
function _nonReentrantAfter() private {
// By storing the original value once again, a refund is triggered (see
// https://eips.ethereum.org/EIPS/eip-2200)
_status = _NOT_ENTERED;
}
}
{
"compilationTarget": {
"contracts/AuctionMarketplace.sol": "AuctionMarketplace"
},
"evmVersion": "paris",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 10000
},
"remappings": [],
"viaIR": true
}
[{"inputs":[],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[],"name":"AuctionExtended","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"BidIndex","type":"uint256"}],"name":"BidReclaimFailed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"BidIndex","type":"uint256"}],"name":"BidReclaimSuccess","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"BidIndex","type":"uint256"}],"name":"BidRefunded","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"BidIndex","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"ETHForBid","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"Unixtimestamp","type":"uint256"},{"indexed":false,"internalType":"address","name":"Bidder","type":"address"}],"name":"BidToppedUp","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"Rebate","type":"uint256"}],"name":"BidTopupRefunded","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"BidIndex","type":"uint256"},{"indexed":false,"internalType":"address","name":"Bidder","type":"address"},{"indexed":false,"internalType":"uint256","name":"MessageValue","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"Unixtimestamp","type":"uint256"},{"indexed":false,"internalType":"address","name":"Vault","type":"address"}],"name":"NewBidComplete","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"Admin","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"AuctionParams","outputs":[{"internalType":"string","name":"_Name","type":"string"},{"internalType":"bool","name":"_SettlementEnabled","type":"bool"},{"internalType":"bool","name":"_UserSettlementEnabled","type":"bool"},{"internalType":"uint256","name":"_LeaderboardSize","type":"uint256"},{"internalType":"uint256","name":"_UnixStartTime","type":"uint256"},{"internalType":"uint256","name":"_UnixEndTime","type":"uint256"},{"internalType":"uint256","name":"_MinBIPSIncrease","type":"uint256"},{"internalType":"uint256","name":"_SecondsExtension","type":"uint256"},{"internalType":"uint256","name":"_SecondsThreshold","type":"uint256"},{"internalType":"uint256","name":"_InitialMinimumBid","type":"uint256"},{"internalType":"uint256","name":"_ProjectIDMintPass","type":"uint256"},{"internalType":"uint256","name":"_MintPackAmount","type":"uint256"},{"internalType":"address","name":"_NFT","type":"address"},{"internalType":"address","name":"_Operator","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"Bids","outputs":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DelegateCash","outputs":[{"internalType":"contract IDelegation","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"Discounts","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256[]","name":"BidIndexes","type":"uint256[]"},{"internalType":"uint256[]","name":"Amounts","type":"uint256[]"}],"name":"IncreaseBid","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"Leaderboard","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"MerkleRootsAmounts","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"MerkleRootsEligibility","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"NFTTokenIDHasBeenSent","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"MaxAmount","type":"uint256"},{"internalType":"address","name":"Vault","type":"address"},{"internalType":"bytes32[]","name":"ProofEligibility","type":"bytes32[]"},{"internalType":"bytes32[]","name":"ProofAmount","type":"bytes32[]"}],"name":"NewBid","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"PriorityPurchaseAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"SaleState","outputs":[{"internalType":"bool","name":"_Active","type":"bool"},{"internalType":"bool","name":"_NFTsDispersed","type":"bool"},{"internalType":"uint256","name":"_LastMinBid","type":"uint256"},{"internalType":"uint256","name":"_GlobalUniqueBids","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"UserBidIndexes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"UserSettleAuction","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewAuctionParams","outputs":[{"components":[{"internalType":"string","name":"_Name","type":"string"},{"internalType":"bool","name":"_Active","type":"bool"},{"internalType":"bool","name":"_NFTsDispersed","type":"bool"},{"internalType":"bool","name":"_SettlementEnabled","type":"bool"},{"internalType":"uint256","name":"_LeaderboardSize","type":"uint256"},{"internalType":"uint256","name":"_UnixStartTime","type":"uint256"},{"internalType":"uint256","name":"_UnixEndTime","type":"uint256"},{"internalType":"uint256","name":"_MinBIPSIncrease","type":"uint256"},{"internalType":"uint256","name":"_SecondsExtension","type":"uint256"},{"internalType":"uint256","name":"_SecondsThreshold","type":"uint256"},{"internalType":"uint256","name":"_LastMinBid","type":"uint256"},{"internalType":"uint256","name":"_GlobalUniqueBids","type":"uint256"},{"internalType":"uint256","name":"_ProjectIDMintPass","type":"uint256"},{"internalType":"uint256","name":"_MintPackAmount","type":"uint256"},{"internalType":"address","name":"_NFT","type":"address"}],"internalType":"struct AuctionMarketplace.AllAuctionParams","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256[]","name":"Indexes","type":"uint256[]"}],"name":"ViewBidsAtIndexes","outputs":[{"components":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"internalType":"struct AuctionMarketplace.Bid[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewBidsUnique","outputs":[{"components":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"internalType":"struct AuctionMarketplace.Bid[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"address","name":"Wallet","type":"address"}],"name":"ViewFrontend","outputs":[{"internalType":"uint256","name":"_LLB","type":"uint256"},{"internalType":"uint256","name":"_MVB","type":"uint256"},{"components":[{"internalType":"string","name":"_Name","type":"string"},{"internalType":"bool","name":"_Active","type":"bool"},{"internalType":"bool","name":"_NFTsDispersed","type":"bool"},{"internalType":"bool","name":"_SettlementEnabled","type":"bool"},{"internalType":"uint256","name":"_LeaderboardSize","type":"uint256"},{"internalType":"uint256","name":"_UnixStartTime","type":"uint256"},{"internalType":"uint256","name":"_UnixEndTime","type":"uint256"},{"internalType":"uint256","name":"_MinBIPSIncrease","type":"uint256"},{"internalType":"uint256","name":"_SecondsExtension","type":"uint256"},{"internalType":"uint256","name":"_SecondsThreshold","type":"uint256"},{"internalType":"uint256","name":"_LastMinBid","type":"uint256"},{"internalType":"uint256","name":"_GlobalUniqueBids","type":"uint256"},{"internalType":"uint256","name":"_ProjectIDMintPass","type":"uint256"},{"internalType":"uint256","name":"_MintPackAmount","type":"uint256"},{"internalType":"address","name":"_NFT","type":"address"}],"internalType":"struct AuctionMarketplace.AllAuctionParams","name":"_AuctionParams","type":"tuple"},{"components":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"internalType":"struct AuctionMarketplace.Bid[]","name":"_RankedLeaderboard","type":"tuple[]"},{"components":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"internalType":"struct AuctionMarketplace.Bid[]","name":"_MasterBids","type":"tuple[]"},{"internalType":"uint256[]","name":"_UserBidIndexes","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewLeaderboardBids","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewLeaderboardIndexes","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewLeaderboardRanked","outputs":[{"components":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"internalType":"struct AuctionMarketplace.Bid[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewMinimumValidBid","outputs":[{"internalType":"uint256","name":"ValidBid","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"ViewRoots","outputs":[{"internalType":"bytes32[]","name":"","type":"bytes32[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"address","name":"Wallet","type":"address"}],"name":"ViewWalletBidIndexes","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"address","name":"Wallet","type":"address"}],"name":"ViewWalletBids","outputs":[{"components":[{"internalType":"uint256","name":"_OGBidIndex","type":"uint256"},{"internalType":"uint256","name":"_Priority","type":"uint256"},{"internalType":"uint256","name":"_Allocation","type":"uint256"},{"internalType":"uint256","name":"_ETHValue","type":"uint256"},{"internalType":"uint256","name":"_Timestamp","type":"uint256"},{"internalType":"address","name":"_Bidder","type":"address"},{"internalType":"address","name":"_Vault","type":"address"},{"internalType":"bool","name":"_Rebated","type":"bool"},{"internalType":"bool","name":"_Winner","type":"bool"},{"internalType":"bool","name":"_NFTSent","type":"bool"}],"internalType":"struct AuctionMarketplace.Bid[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"_GLOBAL_UNIQUE_SALES","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"string","name":"_Name","type":"string"},{"internalType":"bool","name":"_SettlementEnabled","type":"bool"},{"internalType":"bool","name":"_UserSettlementEnabled","type":"bool"},{"internalType":"uint256","name":"_LeaderboardSize","type":"uint256"},{"internalType":"uint256","name":"_UnixStartTime","type":"uint256"},{"internalType":"uint256","name":"_UnixEndTime","type":"uint256"},{"internalType":"uint256","name":"_MinBIPSIncrease","type":"uint256"},{"internalType":"uint256","name":"_SecondsExtension","type":"uint256"},{"internalType":"uint256","name":"_SecondsThreshold","type":"uint256"},{"internalType":"uint256","name":"_InitialMinimumBid","type":"uint256"},{"internalType":"uint256","name":"_ProjectIDMintPass","type":"uint256"},{"internalType":"uint256","name":"_MintPackAmount","type":"uint256"},{"internalType":"address","name":"_NFT","type":"address"},{"internalType":"address","name":"_Operator","type":"address"}],"internalType":"struct AuctionMarketplace.Params","name":"AuctionInfo","type":"tuple"},{"internalType":"bytes32[]","name":"RootsEligibility","type":"bytes32[]"},{"internalType":"bytes32[]","name":"RootsAmounts","type":"bytes32[]"},{"internalType":"uint256[]","name":"DiscountAmounts","type":"uint256[]"}],"name":"__StartAuction","outputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"___ChangeActiveState","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256[]","name":"NewDiscountAmounts","type":"uint256[]"}],"name":"___ChangeDiscountAmounts","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"NewLeaderboardSize","type":"uint256"}],"name":"___ChangeLeaderboardSize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"NewMinBIPSIncrease","type":"uint256"}],"name":"___ChangeMinBIPSIncrease","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"NewMinBid","type":"uint256"}],"name":"___ChangeMinBid","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"MintPassProjectID","type":"uint256"}],"name":"___ChangeMintPassProjectID","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"address","name":"NewAddress","type":"address"}],"name":"___ChangeNFTAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"address","name":"Operator","type":"address"}],"name":"___ChangeOperator","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"bytes32[]","name":"RootsEligibility","type":"bytes32[]"},{"internalType":"bytes32[]","name":"RootsAmounts","type":"bytes32[]"}],"name":"___ChangeRoots","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"Seconds","type":"uint256"}],"name":"___ChangeSecondsExtension","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"Seconds","type":"uint256"}],"name":"___ChangeSecondsThreshold","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"bool","name":"NewState","type":"bool"}],"name":"___ChangeSettlementEnabled","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"NewUnixEndTime","type":"uint256"}],"name":"___ChangeUnixEndTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256","name":"NewUnixStartTime","type":"uint256"}],"name":"___ChangeUnixStartTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"___InitiateRebateAndProceeds","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"___ProcessETHAndNFTsMintPacksRanked","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"___ProcessETHAndNFTsTokenIDsAmbiguous","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"___ProcessETHAndNFTsTokenIDsDistinct","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256[]","name":"TokenIDs","type":"uint256[]"}],"name":"___ProcessETHAndNFTsTokenIDsSpecificRanked","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"},{"internalType":"uint256[]","name":"TokenIDs","type":"uint256[]"}],"name":"___ProcessETHAndNFTsTokenIDsSpecificUnranked","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"Wallet","type":"address"}],"name":"____AuthorizeAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"Wallet","type":"address"}],"name":"____DeuthorizeAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"SaleIndex","type":"uint256"}],"name":"____InitiateOnlyProceeds","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"____WithdrawEther","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address payable","name":"Recipient","type":"address"},{"internalType":"uint256","name":"Amount","type":"uint256"}],"name":"____WithdrawEtherToAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]