Source Code

;; Interface definitions
(impl-trait 'SP2PABAF9FTAJYNFZH93XENAJ8FVY99RRM50D2JG9.nft-trait.nft-trait)
(impl-trait 'SPGAKH27HF1T170QET72C727873H911BKNMPF8YB.operable.operable)

(use-trait commission-trait 'SP3QSAJQ4EA8WXEDSRRKMZZ29NH91VZ6C5X88FGZQ.commission-trait.commission)

;; contract variables
(define-data-var mint-counter uint u1)

(define-data-var token-uri (string-ascii 246) "https://www.stackboard.art/collection/crashpunks/metadata/")
(define-data-var metadata-frozen bool false)

;; constants
(define-constant MINT-PRICE u100000000)
(define-constant XL-MINT-PRICE u360000000)

(define-constant CONTRACT-OWNER contract-caller)
(define-constant token-name "crashpunks-board-slot")
(define-constant COLLECTION-MAX-SUPPLY u56)

(define-constant ERR-METADATA-FROZEN (err u101))
(define-constant ERR-COULDNT-GET-NFT-OWNER (err u102))
(define-constant ERR-PRICE-WAS-ZERO (err u103))
(define-constant ERR-NFT-NOT-LISTED-FOR-SALE (err u104))
(define-constant ERR-NFT-LISTED (err u105))
(define-constant ERR-COLLECTION-LIMIT-REACHED (err u106))
(define-constant ERR-WRONG-COMMISSION (err u107))
(define-constant ERR-ALREADY-MINTED (err u108))
(define-constant ERR-COULDNT-GET-V1-OWNER (err u109))
(define-constant ERR-NOT-V1-OWNER (err u110))
(define-constant ERR-COULDNT-UNWRAP-ID (err u111))
(define-constant ERR-CANT-MINT-ID-ZERO (err u112))

(define-constant ERR-NOT-AUTHORIZED (err u401))
(define-constant ERR-NOT-OWNER (err u402))
(define-constant ERR-NOT-ADMINISTRATOR (err u403))
(define-constant ERR-NOT-FOUND (err u404))

(define-constant wallet-1 'SPGAKH27HF1T170QET72C727873H911BKNMPF8YB)
(define-constant wallet-2 'SP2CBFWG9AT8W4WSCSSJE1R42SDECK7K7W9VSEKD0)
(define-constant wallet-3 'SPZRAE52H2NC2MDBEV8W99RFVPK8Q9BW8H88XV9N)
(define-constant wallet-4 'SP2FTZQX1V9FPPNH485Z49JE914YNQYGT4XVGNR4S)
(define-constant wallet-5 'SP162D87CY84QVVCMJKNKGHC7GGXFGA0TAR9D0XJW)
(define-constant wallet-6 'SP1P89TEC03E29V5MYJBSCC8KWR1A243ZG2R8DYB1)
(define-constant wallet-7 'SP2S6MCR2K3TYAC02RSYQ74RE9RJ3Q0EV3FYFGKGB)

(define-non-fungible-token crashpunks-board-slot uint)

;; data structures
;; slot id -> data hash
(define-map slots-map 
    uint
    (buff 40)
)

;; {owner, operator, id} -> boolean
;; if {owner, operator, id}->true in map, then operator can perform actions on behalf of owner for this id
(define-map approvals {owner: principal, operator: principal, id: uint} bool)
(define-map approvals-all {owner: principal, operator: principal} bool)

;; id -> {price (in ustx), commission trait}
;; if id is not in map, it is not listed for sale
(define-map market uint {price: uint, commission: principal})

;; whitelist address -> # they can mint
(define-map mint-pass principal uint)

;; SIP-09: get last token id
(define-read-only (get-last-token-id)
  (ok (- (var-get mint-counter) u1))
)

;; SIP-09: URI for metadata associated with the token
(define-read-only (get-token-uri (id uint))
    (ok (some (var-get token-uri)))
)

;; SIP-09: Gets the owner of the 'Specified token ID.
(define-read-only (get-owner (id uint))
  (ok (nft-get-owner? crashpunks-board-slot id))
)

;; SIP-09: Transfer
(define-public (transfer (id uint) (owner principal) (recipient principal))
    (begin
        (asserts! (unwrap! (is-approved id contract-caller) ERR-NOT-AUTHORIZED) ERR-NOT-AUTHORIZED)
        (asserts! (is-none (map-get? market id)) ERR-NFT-LISTED)
        (nft-transfer? crashpunks-board-slot id owner recipient)
    )
)

;; operable
(define-read-only (is-approved (id uint) (operator principal))
    (let ((owner (unwrap! (nft-get-owner? crashpunks-board-slot id) ERR-COULDNT-GET-NFT-OWNER)))
        (ok (is-owned-or-approved id operator owner))
    )
)

;; operable
(define-public (set-approved (id uint) (operator principal) (approved bool))
    (ok (map-set approvals {owner: contract-caller, operator: operator, id: id} approved))
)

(define-public (set-approved-all (operator principal) (approved bool))
    (ok (map-set approvals-all {owner: contract-caller, operator: operator} approved))
)

;; public methods
(define-public (mint
    (id uint)
    (for-sale bool)
    (price uint)
    (comm <commission-trait>)
    (data-hash (buff 40))
    (owned-stacksboard-nft-id (optional uint))
    )
    (let (
            (mintCounter (var-get mint-counter))
            (mintPrice (try! (get-mint-price id owned-stacksboard-nft-id)))
        )
        ;; assert we didn't mint more than the limit
        (asserts! (<= mintCounter COLLECTION-MAX-SUPPLY) ERR-COLLECTION-LIMIT-REACHED)

        ;; assert we didn't already mint the slot at this tier and index
        (asserts! (is-eq none (map-get? slots-map id)) ERR-ALREADY-MINTED)
        
        ;; can't mint id 0
        (asserts! (not (is-eq u0 id)) ERR-CANT-MINT-ID-ZERO)

        ;; (try! (stx-transfer? mintPrice contract-caller CONTRACT-OWNER))
        (if (is-eq contract-caller CONTRACT-OWNER)
            true
            (try! (pay-mint mintPrice))
        )

        (try! (nft-mint? crashpunks-board-slot id contract-caller))

        (var-set mint-counter (+ mintCounter u1))
        (map-set slots-map id data-hash)

        (if for-sale 
            (try! (list-in-ustx id price comm))
            (try! (unlist-in-ustx id))
        )

        (ok true)
    )
)

(define-private (pay-mint (mintPrice uint))
    (begin
        (try! (stx-transfer? (/ (* mintPrice u7) u10) contract-caller wallet-1))
        (try! (stx-transfer? (/ (* mintPrice u24) u100) contract-caller wallet-2))
        (try! (stx-transfer? (/ (* mintPrice u108) u10000) contract-caller wallet-1))
        (try! (stx-transfer? (/ (* mintPrice u108) u10000) contract-caller wallet-3))
        (try! (stx-transfer? (/ (* mintPrice u108) u10000) contract-caller wallet-4))
        (try! (stx-transfer? (/ (* mintPrice u108) u10000) contract-caller wallet-5))
        (try! (stx-transfer? (/ (* mintPrice u6) u1000) contract-caller wallet-6))
        (try! (stx-transfer? (/ (* mintPrice u3 ) u4000) contract-caller wallet-7))
        (ok true)
    )
)

(define-public (update-slot 
    (id uint) 
    (for-sale bool)
    (price uint)
    (comm <commission-trait>)
    (data-hash (buff 40))
    )
    (let ((owner (unwrap! (nft-get-owner? crashpunks-board-slot id) ERR-COULDNT-GET-NFT-OWNER)))

        ;; asserts the owner is the contract-caller
        (asserts! (is-eq owner contract-caller) ERR-NOT-OWNER)

        (if for-sale 
            (try! (list-in-ustx id price comm))
            (try! (unlist-in-ustx id))
        )

        (ok (map-set slots-map id data-hash))
    )
)

(define-public (purchase-and-update-slot
    (id uint)
    (for-sale bool)
    (price uint)
    (comm <commission-trait>)
    (data-hash (buff 40))
    )
    (begin 
        (try! (buy-in-ustx id comm))
        (try! (update-slot id for-sale price comm data-hash))
        (ok true)
    )
)

;; marketplace function
(define-public (list-in-ustx (id uint) (price uint) (comm <commission-trait>))
    (let ((listing {price: price, commission: (contract-of comm)})) 
        (asserts! (is-eq contract-caller (unwrap! (nft-get-owner? crashpunks-board-slot id) ERR-COULDNT-GET-NFT-OWNER)) ERR-NOT-OWNER)
        (asserts! (> price u0) ERR-PRICE-WAS-ZERO)
        (ok (map-set market id listing))
    )
)

;; marketplace function
(define-public (unlist-in-ustx (id uint))
    (begin 
        (asserts! (is-eq contract-caller (unwrap! (nft-get-owner? crashpunks-board-slot id) ERR-COULDNT-GET-NFT-OWNER)) ERR-NOT-OWNER)
        (ok (map-delete market id))
    )
)

;; marketplace function
(define-public (buy-in-ustx (id uint) (comm <commission-trait>))
    (let 
        (
            (listing (unwrap! (map-get? market id) ERR-NFT-NOT-LISTED-FOR-SALE))
            (owner (unwrap! (nft-get-owner? crashpunks-board-slot id) ERR-COULDNT-GET-NFT-OWNER))
            (buyer contract-caller)
            (price (get price listing))
        )
        (asserts! (is-eq (contract-of comm) (get commission listing)) ERR-WRONG-COMMISSION)
        (try! (stx-transfer? price contract-caller owner))
        (try! (contract-call? comm pay id price))
        (try! (nft-transfer? crashpunks-board-slot id owner buyer))
        (map-delete market id)
        (ok true)
    )
)

(define-public (burn (id uint))
    (let ((owner (unwrap! (nft-get-owner? crashpunks-board-slot id) ERR-COULDNT-GET-NFT-OWNER)))
        (asserts! (is-eq owner contract-caller) ERR-NOT-OWNER)
        (map-delete market id)
        (map-set slots-map id 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000)
        (nft-burn? crashpunks-board-slot id contract-caller)
    )
)

(define-public (set-token-uri (new-token-uri (string-ascii 80)))
    (begin
        (asserts! (is-eq contract-caller CONTRACT-OWNER) ERR-NOT-ADMINISTRATOR)
        (asserts! (not (var-get metadata-frozen)) ERR-METADATA-FROZEN)
        (var-set token-uri new-token-uri)
        (ok true))
)

(define-public (freeze-metadata)
    (begin
        (asserts! (is-eq contract-caller CONTRACT-OWNER) ERR-NOT-ADMINISTRATOR)
        (var-set metadata-frozen true)
        (ok true)
    )
)

;; read only methods
(define-read-only (get-listing-in-ustx (id uint))
    (map-get? market id)
)

(define-read-only (get-slot-info (id uint))
    (map-get? slots-map id)
)

;; private methods
(define-private (is-owned-or-approved (id uint) (operator principal) (owner principal))
    (default-to 
        (default-to
            (is-eq owner operator)
            (map-get? approvals-all {owner: owner, operator: operator})
        )
        (map-get? approvals {owner: owner, operator: operator, id: id})
    )
)

(define-private (get-base-mint-price (id uint))
    (if (<= id u8)
        XL-MINT-PRICE
        MINT-PRICE
    )
)

(define-private (get-mint-price (id uint) (stacksboard-nft-id (optional uint)))
    (let ((mintPrice (get-base-mint-price id)))
        (if (is-none stacksboard-nft-id)
            (ok mintPrice)
            (let (
                    (unwrapped-stacksboard-nft-id (unwrap! stacksboard-nft-id ERR-COULDNT-UNWRAP-ID))
                    (stacksboard-owner (unwrap! (unwrap! (contract-call? 'SPGAKH27HF1T170QET72C727873H911BKNMPF8YB.stacks-board-slot get-owner unwrapped-stacksboard-nft-id) ERR-COULDNT-GET-V1-OWNER) ERR-COULDNT-GET-V1-OWNER))
                )
                (asserts! (is-eq stacksboard-owner contract-caller) ERR-NOT-V1-OWNER)
                (if (<= unwrapped-stacksboard-nft-id u56)
                    (ok (/ (* mintPrice u75) u100))
                    (ok (/ (* mintPrice u90) u100))
                )
            )
        )
    )
)

(begin
    (try! (mint u41 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u42 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u43 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u44 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u45 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u46 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u47 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u48 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u49 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u50 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u51 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u52 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u53 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u54 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u55 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
    (try! (mint u56 false u0 .commission-stacksboard 0x00000000000000000000000000000000000000000000000000000000000000000000000000000000 none))
)

Functions (22)

FunctionAccessArgs
get-last-token-idread-only
get-token-uriread-onlyid: uint
get-ownerread-onlyid: uint
transferpublicid: uint, owner: principal, recipient: principal
is-approvedread-onlyid: uint, operator: principal
set-approvedpublicid: uint, operator: principal, approved: bool
set-approved-allpublicoperator: principal, approved: bool
mintpublicid: uint, for-sale: bool, price: uint, comm: <commission-trait>, data-hash: (buff 40
pay-mintprivatemintPrice: uint
update-slotpublicid: uint, for-sale: bool, price: uint, comm: <commission-trait>, data-hash: (buff 40
purchase-and-update-slotpublicid: uint, for-sale: bool, price: uint, comm: <commission-trait>, data-hash: (buff 40
list-in-ustxpublicid: uint, price: uint, comm: <commission-trait>
unlist-in-ustxpublicid: uint
buy-in-ustxpublicid: uint, comm: <commission-trait>
burnpublicid: uint
set-token-uripublicnew-token-uri: (string-ascii 80
freeze-metadatapublic
get-listing-in-ustxread-onlyid: uint
get-slot-inforead-onlyid: uint
is-owned-or-approvedprivateid: uint, operator: principal, owner: principal
get-base-mint-priceprivateid: uint
get-mint-priceprivateid: uint, stacksboard-nft-id: (optional uint