Source Code

;; Stacks Art NFTs
;; Used for auctions and minting single NFTs
(impl-trait 'SP2PABAF9FTAJYNFZH93XENAJ8FVY99RRM50D2JG9.nft-trait.nft-trait)
(define-non-fungible-token stacks-art uint)

;; constants
(define-constant ERR-NOT-AUTHORIZED u401)

;; variables
(define-constant LIST_5 (list
  true true true true true
))
(define-constant LIST_10 (list
  true true true true true true true true true true
))
(define-constant LIST_20 (list
  true true true true true true true true true true
  true true true true true true true true true true
))
(define-constant LIST_30 (list
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
))
(define-constant LIST_40 (list
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
))
(define-constant LIST_50 (list
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
))
(define-constant LIST_100 (list
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
  true true true true true true true true true true
))

(define-data-var CONTRACT-OWNER principal tx-sender)
(define-data-var last-id uint u1)
(define-data-var last-series-id uint u1)
(define-map metadata { id: uint } {
  amount: uint,                       ;; amount of NFTs exist (default 1)
  series-id: uint,                    ;; ID in the series
  series-index: uint,                 ;; index in the series (if amount > 1)
  creator: principal,                 ;; creator address
  name: (string-utf8 256),            ;; name of the NFT
  image: (string-ascii 256),          ;; IPFS/AR image URI
  uri: (string-ascii 256),            ;; IPFS/AR metadata URI
  mime-type: (string-ascii 256),      ;; JPG/GIF/MP4/...
  hash: (buff 64),                    ;; unique hash of the file
  frozen: bool                        ;; entry can be changed or not
})

(define-read-only (get-metadata (id uint))
  (default-to
    { amount: u0, series-id: u0, series-index: u0, creator: (var-get CONTRACT-OWNER), name: u"", image: "", uri: "", mime-type: "", hash: 0x00, frozen: false }
    (map-get? metadata { id: id })
  )
)

(define-public (mint
  (amount uint)
  (series-index uint)
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
  (frozen bool)
)
  (let (
    (id (var-get last-id))
  )
    (try! (mint-internal (var-get last-series-id) amount series-index creator name image uri mime-type hash frozen))
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok id)
  )
)

(define-private (mint-internal
  (series-id uint)
  (amount uint)
  (series-index uint)
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
  (frozen bool)
)
  (let (
    (id (var-get last-id))
  )
    (map-set metadata { id: id } {
      amount: amount,
      series-id: series-id,
      series-index: series-index,
      creator: creator,
      name: name,
      image: image,
      uri: uri,
      mime-type: mime-type,
      hash: hash,
      frozen: frozen
    })
    (try! (nft-mint? stacks-art id tx-sender))
    (var-set last-id (+ id u1))
    (ok id)
  )
)

;; Mint a series of NFTs
(define-private (mint-n
  (i bool)
  (data
    {
      amount: uint,
      series-index: uint,
      creator: principal,
      name: (string-utf8 256),
      image: (string-ascii 256),
      uri: (string-ascii 256),
      mime-type: (string-ascii 256),
      hash: (buff 64)
    }
  )
)
  (begin
    (asserts!
      (is-ok (mint-internal
          (var-get last-series-id)
          (get amount data)
          (+ u1 (get series-index data))
          (get creator data)
          (get name data)
          (get image data)
          (get uri data)
          (get mime-type data)
          (get hash data)
          true
        )
      )
      data
    )
    (merge data { series-index: (+ u1 (get series-index data)) })
  )
)

(define-public (mint-custom
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
  (amount uint)
  (iterator (list 200 bool))
)
  (let (
    (result
      (fold mint-n iterator
        { amount: amount, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (mint-five
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_5
        { amount: u5, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (mint-ten
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_10
        { amount: u10, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (mint-twenty
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_20
        { amount: u20, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (mint-thirty
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_30
        { amount: u30, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (mint-forty
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_40
        { amount: u40, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (ok true)
  )
)

(define-public (mint-fifty
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_50
        { amount: u50, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (mint-one-hundred
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (result
      (fold mint-n LIST_100
        { amount: u100, series-index: u0, creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }
      )
    )
  )
    (print result)
    (var-set last-series-id (+ (var-get last-series-id) u1))
    (ok true)
  )
)

(define-public (update-metadata
  (id uint)
  (creator principal)
  (name (string-utf8 256))
  (image (string-ascii 256))
  (uri (string-ascii 256))
  (mime-type (string-ascii 256))
  (hash (buff 64))
)
  (let (
    (data (get-metadata id))
  )
    (asserts! (is-eq contract-caller (get creator data)) (err ERR-NOT-AUTHORIZED))
    (asserts! (not (get frozen data)) (err ERR-NOT-AUTHORIZED))

    (map-set metadata { id: id } (merge data { creator: creator, name: name, image: image, uri: uri, mime-type: mime-type, hash: hash }))
    (ok true)
  )
)

(define-public (freeze-metadata (id uint))
  (let (
    (data (get-metadata id))
  )
    (asserts!
      (or
        (is-eq contract-caller (get creator data))
        (is-eq contract-caller (var-get CONTRACT-OWNER))
      )
      (err ERR-NOT-AUTHORIZED)
    )

    (map-set metadata { id: id } (merge data { frozen: true }))
    (ok true)
  )
)

(define-public (transfer (index uint) (owner principal) (recipient principal))
  (if (and (is-owner index owner) (is-owner index tx-sender))
    (match (nft-transfer? stacks-art index owner recipient)
      success (ok true)
      error (err error)
    )
    (err ERR-NOT-AUTHORIZED)
  )
)

(define-public (burn (id uint))
  (begin
    (asserts! (is-owner id tx-sender) (err ERR-NOT-AUTHORIZED))
    (nft-burn? stacks-art id tx-sender)
  )
)

(define-read-only (get-owner (id uint))
  (ok (nft-get-owner? stacks-art id))
)

(define-read-only (get-last-token-id)
  (ok (var-get last-id))
)

(define-read-only (get-image-uri (id uint))
  (ok (some (get image (get-metadata id))))
)

(define-read-only (get-token-uri (id uint))
  (ok (some (get uri (get-metadata id))))
)

(define-private (is-owner (index uint) (user principal))
  (is-eq user (unwrap! (nft-get-owner? stacks-art index) false))
)

Functions (20)

FunctionAccessArgs
get-metadataread-onlyid: uint
mintpublicamount: uint, series-index: uint, creator: principal, name: (string-utf8 256
mint-internalprivateseries-id: uint, amount: uint, series-index: uint, creator: principal, name: (string-utf8 256
mint-custompubliccreator: principal, name: (string-utf8 256
mint-fivepubliccreator: principal, name: (string-utf8 256
mint-tenpubliccreator: principal, name: (string-utf8 256
mint-twentypubliccreator: principal, name: (string-utf8 256
mint-thirtypubliccreator: principal, name: (string-utf8 256
mint-fortypubliccreator: principal, name: (string-utf8 256
mint-fiftypubliccreator: principal, name: (string-utf8 256
mint-one-hundredpubliccreator: principal, name: (string-utf8 256
update-metadatapublicid: uint, creator: principal, name: (string-utf8 256
freeze-metadatapublicid: uint
transferpublicindex: uint, owner: principal, recipient: principal
burnpublicid: uint
get-ownerread-onlyid: uint
get-last-token-idread-only
get-image-uriread-onlyid: uint
get-token-uriread-onlyid: uint
is-ownerprivateindex: uint, user: principal