English 简体中文 繁體中文 한국 사람 日本語 Deutsch русский بالعربية TÜRKÇE português คนไทย french

简体中文 繁體中文 English 日本語 Deutsch 한국 사람 بالعربية TÜRKÇE คนไทย Français русский

回答

收藏

7.5 NFT 项目/SBT 项目智能合约

开源社区 开源社区 7003 人阅读 | 0 人回复 | 2025-03-15

在本课中,我们将回顾经典的 NFT 项目的代码。此外,我们还将学习 SoulBound (SBT) Item。

7 c6 A5 c  X% p+ O+ f7 w) PNFT Item合约
让我们打开并查看 contract.
  1. int min_tons_for_storage() asm "50000000 PUSHINT"; ;; 0.05 TON
    & |2 U2 A, x* n/ H
复制代码
就像在前面的课程中一样,我们看到有一些常量显示了我们应该在合约上存储的最低代币数量,以便能够支付租金。
- n2 J9 H# z6 \# Z% ^
  1. (int, int, slice, slice, cell) load_data() {
    * v+ R$ s1 [4 d0 I  S+ z
  2.     slice ds = get_data().begin_parse();% S( K% @5 o( C  v7 s; v: p  @
  3.     var (index, collection_address) = (ds~load_uint(64), ds~load_msg_addr());
    + h& F4 f0 Q& L, P0 R3 N: d1 [
  4.     if (ds.slice_bits() > 0) {; I/ @. K4 C: C1 F
  5.       return (-1, index, collection_address, ds~load_msg_addr(), ds~load_ref());
    ; N9 S0 A% N% |
  6.     } else {  " s" ]" g, T. o1 Z0 `! ]
  7.       return (0, index, collection_address, null(), null()); ;; nft not initialized yet% f, S8 I3 I/ a0 M% [
  8.     }
    # y( Q2 F5 v, a
  9. }
复制代码
然后我们看到 load_data. 这个 load_data 与我们以前看到的有些不同。通常,我们只是从本地存储区读取数据,但在这里,我们有一些更复杂的逻辑。因此,我们打开本地存储的Cell进行读取,然后读出 index 和 collection_address. 我们知道什么是索引和 Collections 地址。  
然后,我们尝试读取 NFT Item的元数据。首先,我们检查它是否存在。如果是,我们返回 –1, 以及Collections地址、所有者地址和元数据。或者我们只返回 0, 索引和Collections地址时,没有所有者和元数据。这意味着 NFT 尚未初始化。基于这一点,我们将在后面的代码中做出一些决定。
  1. () store_data(int index, slice collection_address, slice owner_address, cell content) impure {) e" [. |+ ]! p% i2 o
  2.     set_data(8 l, D$ H$ e3 r7 L
  3.         begin_cell()/ J7 E6 [# v4 [+ n* U
  4.             .store_uint(index, 64)
    # x$ }5 u# u! w
  5.             .store_slice(collection_address)
    8 d: b& l6 T0 E& ^# G, r
  6.             .store_slice(owner_address)4 ?0 C7 s4 f6 P( s$ U, y5 d/ _
  7.             .store_ref(content)
    1 w# O, g/ [+ `6 V; O- p+ g% H  X
  8.             .end_cell()
    $ ?) }4 {$ q; {1 ]5 C/ t" o
  9.     );
    ( Y; u$ J( W4 m
  10. }
复制代码
我们还看到 store_data 就像我们调查的其他合约一样,这只是正常情况。

, |( f. r& }' s& a3 h
  1. () send_msg(slice to_address, int amount, int op, int query_id, builder payload, int send_mode) impure inline {
    & D- L: H2 f3 c% D5 R% B
  2.   var msg = begin_cell()
      L0 n! }  w* {
  3.     .store_uint(0x10, 6) ;; nobounce - int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool src:MsgAddress -> 010000
    7 T7 \- S0 x. s. h% x) c
  4.     .store_slice(to_address). D& Y* t8 J, y
  5.     .store_coins(amount); Y: P/ x2 p+ x& Q6 R
  6.     .store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1)
    5 u8 D) j* Q* m5 P2 w/ w
  7.     .store_uint(op, 32)9 \/ G) s' K  n
  8.     .store_uint(query_id, 64);) S  _6 [. D3 h4 p& \1 F0 s' h

  9. / {8 m8 y+ S' }9 z# p/ F
  10.   if (~ builder_null?(payload)) {  U; `  N0 [. O0 ?' b" @: X
  11.     msg = msg.store_builder(payload);
    3 r9 ~# D! Z7 l+ n% b
  12.   }
复制代码
在这里,我们封装了发送消息的逻辑,因此我们可以在合约的其他逻辑中使用这个函数。因此,只要我们想发送信息,就可以使用这个函数。它接受目标地址、我们应附加的金额、操作代码、查询 ID、可能的有效载荷(这次是以生成器格式)以及消息的发送模式。这就是我们发送信息的逻辑。
我们的另一个函数是 transfer_ownership. 让我们稍后再研究,一旦我们在 recv_internal 逻辑,因为它是这里最重要的功能。
  1. () recv_internal(int my_balance, int msg_value, cell in_msg_full, slice in_msg_body) impure {
    % Z1 s: ~) o& `& g
  2.     if (in_msg_body.slice_empty?()) { ;; ignore empty messages' x% E. `' N7 ^( J
  3.         return ();& d  \) ~" l: z
  4.     }$ G' S; Y% B9 w8 r4 u) }  k; X# u

  5. 1 I8 h2 p3 f& N3 Z( e! A
  6.     slice cs = in_msg_full.begin_parse();- p, ?" q8 _2 J/ j7 F& B6 c/ E
  7.     int flags = cs~load_uint(4);2 Z- {' K0 o) j
  8. ; D5 [) d) }: ?$ D$ k
  9.     if (flags & 1) { ;; ignore all bounced messages
    ( W0 g+ ]' j7 C. }, w
  10.         return ();
    % K/ I5 r, e# @) c/ ]5 t
  11.     }4 {7 s% V& y9 u' a) J
  12.     slice sender_address = cs~load_msg_addr();
    + H& _  A+ }" K% R

  13. 7 T/ O- C) _9 x8 E# ?* d! r
  14.     cs~load_msg_addr(); ;; skip dst( P# z- z0 y+ r+ o9 J: \4 ^& N, S
  15.     cs~load_coins(); ;; skip value
    + ?; Y# k( r. X, e4 G, ?
  16.     cs~skip_bits(1); ;; skip extracurrency collection4 r& m2 C4 k& T8 }) i# p
  17.     cs~load_coins(); ;; skip ihr_fee. ~3 j" b- H5 }
  18.     int fwd_fee = muldiv(cs~load_coins(), 3, 2); ;; we use message fwd_fee for estimation of forward_payload costs
    % |6 Z* [7 O" U3 e' o! ?

  19.   P! B5 |& l  l0 A
  20.     (int init?, int index, slice collection_address, slice owner_address, cell content) = load_data();
    0 e% R6 b7 c0 ^# j- P9 V: V
  21.     if (~ init?) {. R/ ~" l8 @3 Z: o. C5 A4 r) `$ P8 D) E
  22.       throw_unless(405, equal_slices(collection_address, sender_address));
    0 Z# u& N; t- w1 \5 L: g
  23.       store_data(index, collection_address, in_msg_body~load_msg_addr(), in_msg_body~load_ref());1 y  J; h4 X" S( x/ Y) T
  24.       return ();; D; g* d0 e; p4 T9 ~  M7 G, o
  25.     }
复制代码
我们来看看它是怎么做的。我们再次忽略空的 in_msg_body 消息,读取标志,忽略退回的信息,并检查发件人是谁。现在我们跳过一些内容,比如信息的目标地址之类的,因为你已经知道这个Cell的内容是什么了。接下来是 fwd_fee 再次根据上一笔交易估算发送下一笔交易的成本。
然后加载数据。正如你所记得的,我们加载数据的方式与这里的实现方式相同。因此,我们要检查它是否未初始化,然后我们期待实际获得这些数据,期待 in_msg_body 以获得所有者的地址和带有元数据的Cell。但这只能通过 collection_address, 因此我们要检查它是否等于发件人地址。然后我们就可以初始化 NFT Item了。
  1. int op = in_msg_body~load_uint(32);' N) I5 q3 w4 ~' @% w
  2.   int query_id = in_msg_body~load_uint(64);
    1 x6 d! @+ \8 e3 Y! w' B; A

  3. - M+ F+ m* Z% c7 {1 C$ n5 g1 H# U
  4.   if (op == op::transfer()) {6 F2 e- [& r  [( f+ e& v2 n
  5.     transfer_ownership(my_balance, index, collection_address, owner_address, content, sender_address, query_id, in_msg_body, fwd_fee);
    & l1 c$ ]" s3 R
  6.     return ();
    : J! d4 D7 ?: U5 r, u. r3 f* k; i
  7.   }
    / z" J, R* O: Q. F1 p
  8.   if (op == op::get_static_data()) {% }7 Y  K$ A4 i( J. e1 K
  9.     send_msg(sender_address, 0, op::report_static_data(), query_id, begin_cell().store_uint(index, 256).store_slice(collection_address), 64);  ;; carry all the remaining value of the inbound message
    0 ~( \, @) `: I* r/ x
  10.     return ();
      \4 B3 E% m- N8 E5 g% b3 {
  11.   }
复制代码
接下来是操作代码,可以是 op::transfer 或 op::get_static_data. 这是我们要处理的两个可能值。当我们说 get_static_data, 我们会立即向发送静态数据请求的用户发送一条信息。我们只是报告数据,所以我们会向他发送 index 和 collection_address. 这是 get_static_data 函数。正如你所看到的,这是我们的第一个 send_ msg 用例
, {8 K5 ?( Y. x2 @  e2 Y" o6 ^
  1. () transfer_ownership(int my_balance, int index, slice collection_address, slice owner_address, cell content, slice sender_address, int query_id, slice in_msg_body, int fwd_fees) impure inline {
    * Z0 o- \, M! d% Z# C9 |% o+ U
  2.     throw_unless(401, equal_slices(sender_address, owner_address));0 l9 ]! k+ N# K# q
  3. 0 I( d% ~6 k7 U' v. S5 _. C1 `
  4.     slice new_owner_address = in_msg_body~load_msg_addr();
      N. B( |( z+ I$ W  h
  5.     force_chain(new_owner_address);( H2 y$ n) x$ ^4 `+ ~
  6.     slice response_destination = in_msg_body~load_msg_addr();5 C& H' C( W4 n. C
  7.     in_msg_body~load_int(1); ;; this nft don't use custom_payload
    $ o5 F/ [7 {! T; Z- K
  8.     int forward_amount = in_msg_body~load_coins();
    3 f# G! }% P- A  I  S  a$ Y6 Q
  9.     throw_unless(708, slice_bits(in_msg_body) >= 1);: o  _! K8 S, J+ d+ A+ R$ Y+ A
  10. ) W7 U% T9 @4 ?% K: G: n
  11.     int rest_amount = my_balance - min_tons_for_storage();
    ! P9 U' S2 Y3 F. H$ r: v7 P
  12.     if (forward_amount) {- J& v" g6 ?7 f8 a% n+ W
  13.       rest_amount -= (forward_amount + fwd_fees);
    2 P- J" S' H  V" y! f9 Y7 {
  14.     }4 _1 W( {% S5 t% b6 J. [
  15.     int need_response = response_destination.preload_uint(2) != 0; ;; if NOT addr_none: 00
    $ C; a* h2 ?$ A1 b& [( ^) ]
  16.     if (need_response) {
    & k" p5 U5 s. i6 A
  17.       rest_amount -= fwd_fees;) z$ J) P& n0 C; |- T' p
  18.     }
    ( I5 Z1 A. D( C

  19. 9 Z6 Z% }9 c7 p
  20.     throw_unless(402, rest_amount >= 0); ;; base nft spends fixed amount of gas, will not check for response
    " s9 L4 r1 n4 q# E' I# ^

  21. 1 J( y& a4 o. O0 x8 k; I0 Q
  22.     if (forward_amount) {
    $ E: j$ N  K% v+ q; ?( h1 Q: i* ~. j
  23.       send_msg(new_owner_address, forward_amount, op::ownership_assigned(), query_id, begin_cell().store_slice(owner_address).store_slice(in_msg_body), 1);  ;; paying fees, revert on errors- Z" f3 w" G$ k: P1 r/ w3 X
  24.     }+ x; Y; K# L) \' N: G! D8 u, ^
  25.     if (need_response) {4 ?) R6 e- `- i/ K) B$ o
  26.       force_chain(response_destination);+ K+ W" }4 e7 Q+ D0 v
  27.       send_msg(response_destination, rest_amount, op::excesses(), query_id, null(), 1); ;; paying fees, revert on errors% H9 k" h7 y: l, C
  28.     }
    ' f3 h% X  O* P/ X

  29. 4 t/ q: Q0 K3 f* _* J* d( f
  30.     store_data(index, collection_address, new_owner_address, content);; ]* D' q: Z7 M) K) K6 O) X5 r
  31. }
复制代码
这里有一个转移所有权的功能。首先,我们要检查物品的所有者地址是否就是发送信息的人,因为只有他才能转移所有权。然后,我们读出新所有者的地址。我们 force_chain 以确保他在同一条链上。然后我们检查是否有 response_destination, 因此,一旦我们转移了这个所有权,我们是否应该告诉某人向另一个人发送信息。我们检查 forward_amount 是多少,如果有人要求的话。然后,我们就可以计算出,在发出信息之后,我们就会有足够的钱 forward_amount 到目的地。然后,我们要弄清楚是否需要回应,如果存在 response_destination 。如果存在,其剩余金额必须大于 0。
下面我就向大家简单介绍一下它的工作原理。如果存在 forward_amount ,我们将发送一条信息,通知用户所有权已由该 forward_amount 分配。如果我们需要回应,我们将 force_chain 以确保响应目的地在同一链上。我们发送的回复通常包括超额费用部分和所有额外资金;它们只是被转发到 response_destination. 然后,我们只需使用 new_owner_address 保存数据。基本上,这是唯一会发生变化的地方、 transfer_ownership 是这里的核心函数之一。您拥有一件物品,有时您会将其所有权转让给其他人或出售,这取决于交易的类型。

1 U  T$ Q, w4 b, t) m7 W
SBT item合约
NFT 可以在另一个层面运作。查看 Getgems (https://getgems.io/) 市场。它们有自己的合约,可以出售这类资产。我们现在不研究这些市场合约。不过,我还是要告诉你一个地方,在那里你可以学到很多关于 NFT 的知识 — GetGems github — 在这里,我还想为大家介绍一种 NFT。
GetGems repository 正在主办一些不同的 NFT 合约、藏品、销售和市场以及一些拍卖活动。这里是了解更多与 NFT 藏品、物品及其相关的合约的好地方。但我想向您展示一份确切的合约,我们可以对其进行深入研究。
sbt-item 是一种灵魂绑定代币(SBT)。灵魂绑定与 NFT 非常相似,但不同之处在于灵魂绑定代币不能擅自转让给其他人。我们仔细看看它的代码。
  1. global int storage::index;
    ) y9 M7 }% c) D* `/ c; `1 ]% Q
  2. global int init?;
    " c6 Q* S8 ?9 Q
  3. global slice storage::collection_address;
    7 w: U# }' d+ J6 T  l( {. W
  4. global slice storage::owner_address;  l3 P3 W1 W4 ^# j- e) H
  5. global slice storage::authority_address;2 E% E" C: [. o
  6. global cell storage::content;1 W- ^1 g3 H4 Y0 B! f2 d+ a
  7. global int storage::revoked_at;7 T' g* D. ^+ t+ b  T
  8. () load_data() impure {# j& K; Q: T5 l1 U
  9.     slice ds = get_data().begin_parse();
    1 D/ V/ g, t# }. `
  10. ( U) z1 c3 I" b4 P$ H, V; N& z
  11.     storage::index              = ds~load_uint(64);
    & x6 d- {# _1 V# p8 U7 a
  12.     storage::collection_address = ds~load_msg_addr();
    % V: G5 l- i4 l
  13.     init?                       = false;
    " Z/ ?( L: Q; S# Q/ Z
  14. / x* u2 R5 T% {3 @: Y9 O+ H
  15.     if (ds.slice_bits() > 0) {9 M& O1 |+ m7 V: s9 Q! F
  16.         init?                      = true;! N" V* @+ Y7 V. r4 r& @! Z
  17.         storage::owner_address     = ds~load_msg_addr();! ^) R9 \+ ~7 k9 j- m
  18.         storage::content           = ds~load_ref();
    ) X, N% `+ F; d* B) Q
  19.         storage::authority_address = ds~load_msg_addr();
    ) E! ~5 y$ ~, Q/ g+ J$ M! N# K
  20.         storage::revoked_at        = ds~load_uint(64);/ h% T1 D( k: `  [: @7 P& h& b
  21.     }
    ( b# ~/ P9 ^) }6 X$ k& k
  22. }
复制代码
这里有一些变量: storage::index, storage::collection_address. 我们还定义了其他全局变量,这只是 GetGems 实现的一个例子。你不必用同样的方法。我想在这里重点谈谈其他部分:

/ m  L+ J% r, y6 Z# o3 F* |& _
  1. if (op == op::request_owner()) {" l( n8 V- g7 @0 N2 d. g! L
  2.     slice dest = in_msg_body~load_msg_addr();
    " Z/ @! d* G+ o$ p3 ~: s! G) m# l
  3.     cell body = in_msg_body~load_ref();7 O7 [( A- I, x% Y7 }$ N
  4.     int with_content = in_msg_body~load_uint(1);9 ^' w' X8 U# C8 J1 |

  5. " _2 a; d: S+ m
  6.     var msg = begin_cell()
    0 v5 P- ~+ B0 R# _
  7.             .store_uint(storage::index, 256)4 P: ^3 N$ w$ G9 J
  8.             .store_slice(sender_address)
    2 J9 X% G' N: t: B
  9.             .store_slice(storage::owner_address)0 R9 D' B) _4 l6 i
  10.             .store_ref(body)
    4 b7 n# D/ ?; K/ K! F
  11.             .store_uint(storage::revoked_at, 64)! |1 J9 C: i4 d0 E( n6 R
  12.             .store_uint(with_content, 1);
    ( e, ?1 p4 x& [+ Z- q8 D2 e( ]
  13. / W6 B- g& Q# n9 T
  14.     if (with_content != 0) {& k$ s+ o1 K# s0 d) f: G' l
  15.         msg = msg.store_ref(storage::content);
    8 z" x# C" l2 k' n# }% D. a; e% s
  16.     }( S1 S- \3 `$ Y5 M* m# f3 ~

  17. 4 i/ ~2 n; ~9 c! L8 W
  18.     ;; mode 64 = carry all the remaining value of the inbound message$ G% W: X; a2 p8 O5 s2 N& f2 t: d
  19.     send_msg(flag::regular() | flag::bounce(), dest, 0, op::owner_info(), query_id, msg, 64);0 `4 C$ k- F  W6 {# V& P' P
  20.     return ();" d( [) S/ ]* {% |$ O( [
  21. }
    ! j5 J/ x7 n1 t  a- ]9 x& H, v
  22. if (op == op::prove_ownership()) {% }1 m" ]. R) R% t
  23.     throw_unless(401, equal_slices(storage::owner_address, sender_address));1 y- G! z$ N* s% b" @: ~  }

  24. + ]( o  w+ I( q! O2 Y- x
  25.     slice dest = in_msg_body~load_msg_addr();
    " e9 K8 r. _& D0 W, K( j
  26.     cell body = in_msg_body~load_ref();7 t5 ?: f& \9 s3 A
  27.     int with_content = in_msg_body~load_uint(1);
    + ^9 B; F2 l) u( K( s
  28. 6 O$ Z+ @$ R& [! W7 t/ @
  29.     var msg = begin_cell(): P8 T1 g0 b% a9 _) q  M$ X
  30.             .store_uint(storage::index, 256)
    3 e& m) J6 H) `1 S* \2 C
  31.             .store_slice(storage::owner_address); e- w5 L# N: U
  32.             .store_ref(body)
    : }" e2 k7 \' O5 z; z3 ?- T& ]
  33.             .store_uint(storage::revoked_at, 64), t/ _8 {3 a; J) Z6 K! ?$ ~
  34.             .store_uint(with_content, 1);/ X3 @0 l' D9 C1 x' M

  35. $ u' H+ e% e  [$ h; U3 t+ \
  36.     if (with_content != 0) {) n/ t" r1 r2 L
  37.         msg = msg.store_ref(storage::content);
    ) U& r5 L2 ^. h* j- p6 I8 J
  38.     }& J5 J4 i& W1 s
  39. * a/ q' n0 ]- j' S
  40.     ;; mode 64 = carry all the remaining value of the inbound message
    0 k4 |/ D# C8 O3 X6 L+ A
  41.     send_msg(flag::regular() | flag::bounce(), dest, 0, op::ownership_proof(), query_id, msg, 64);
    4 _0 p1 x: n4 Z: e4 k
  42.     return ();% n; z/ k) f" p# I/ Q" {0 f
  43. }
    / X# t' H: V  e. n
  44. if (op == op::get_static_data()) {3 l8 e# ]5 _* J% F
  45.     var msg = begin_cell().store_uint(storage::index, 256).store_slice(storage::collection_address);0 V' j; b8 u1 ?' C

  46. ) C/ s7 [; @$ o) o
  47.     ;; mode 64 = carry all the remaining value of the inbound message5 d7 P& e& N  |% U) m7 L8 ?( M$ T# q
  48.     send_msg(flag::regular(), sender_address, 0, op::report_static_data(), query_id, msg, 64);
    ; w# u* I% M" ~# B
  49.     return ();* I% z  A5 K: O
  50. }
    $ R+ l4 g, @- F- a3 P' d
  51. if (op == op::destroy()) {
    * ?  X7 @; x$ R1 z3 c1 j; G
  52.     throw_unless(401, equal_slices(storage::owner_address, sender_address));$ \. h8 g6 }* E" _1 R" s4 J

  53. 5 O7 z' h; }: K' N( W
  54.     send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);
    ) `( [3 R) A" k+ |3 z7 ~$ l
  55. ' m* i$ R( d' Z0 p+ e* @
  56.     storage::owner_address = null_addr();5 I, I: B  S5 T3 R6 F
  57.     storage::authority_address = null_addr();+ F9 l8 A9 E- A1 t" o  |8 _
  58.     store_data();1 Q5 q0 t" f* V& l5 T% ]# x: ]
  59.     return ();( ?# G( p  w: X! R* v; z
  60. }* J5 z; ^3 \) G+ x+ N8 W
  61. if (op == op::revoke()) {
    9 `1 f4 z# D9 t& E
  62.     throw_unless(401, equal_slices(storage::authority_address, sender_address));
    / W! n5 L! L% V
  63.     throw_unless(403, storage::revoked_at == 0);5 C1 |6 [3 ^8 K0 v
  64. , U$ X$ U9 L* T$ @! f
  65.     storage::revoked_at = now();! O9 u% ~! h, x" U
  66.     store_data();. T& I. P, z; l' K- ^$ s- |
  67.     return ();
    " A$ Q3 [' U$ I: K
  68. }
    & H& D* h. z1 @" i
  69. if (op == op::take_excess()) {2 Q+ f1 F+ c: p
  70.     throw_unless(401, equal_slices(storage::owner_address, sender_address));  p1 J& _! g( U% {
  71. ; }* D4 O+ f& |; u3 R3 b- A
  72.     ;; reserve amount for storage
    2 |9 T3 P. }+ ]+ y% K, x4 H" _+ }
  73.     raw_reserve(min_tons_for_storage(), 0);( p; Y) d3 c9 x+ z: @3 s, l3 K8 u6 z! |0 H

  74. ) I$ O9 P0 K2 C  E' V4 V  d
  75.     send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);; ~5 u/ t: @- C. @, c% l& l
  76.     return ();' A) l4 S3 M* s3 ?( Z5 ?+ P
  77. }
    ! M: e) q/ G* b7 R
  78. if (op == op::transfer()) {& [( R" |( g& h: e3 z: o, g; J& r
  79.     throw(413);
    * ~1 U% C+ g* ~9 u3 z
  80. }
复制代码
如你所见,这里有许多注释,但我们不会逐行深入研究。你可以看到,这个通证将被用于多种不同的用途,但与普通的 NFT 不同。这里是合约,你可以申请这个合约的所有者。一些操作代码可以请求所有权证明。您可以获取静态数据。你还可以销毁或撤销这个标记。你可以拿走多余的东西,比如,存放在这里的一些钱。如果你是这个合约的所有者,你就可以申请。但你显然不能转让它,就像你在这里看到的一样。这是一种完全不同类型的代币,如果你对 NFT 感兴趣,就应该在 GetGems 存储库里多花点时间。
我想向你展示的是,NFT collections 的所有代码可以保持不变,但collections合约部署的 item 可以不同。例如,你可以用 SBT item 替换 NFT item。这样您就会有一个 SBT item collection,然后您只需更改  code 来初始化你的 collection:
  1. (slice, int, cell, cell, cell) load_data() inline {
    6 G% B5 D% {0 ^
  2.   var ds = get_data().begin_parse();
    . T5 D7 ]" E  F0 f; W  |
  3.   return
    # `& c- C; \. r$ K' Z
  4.     (ds~load_msg_addr(), ;; owner_address8 z: i& z. z" ?$ O. n" C  |
  5.      ds~load_uint(64), ;; next_item_index
    . F! C5 o3 U  |# a$ ]
  6.      ds~load_ref(), ;; content
    4 O: o  y2 z3 w1 c( F% Z9 {
  7.      ds~load_ref(), ;; nft_item_code
    3 Y$ c/ T  K% e4 a7 O. {
  8.      ds~load_ref()  ;; royalty_params
    5 l% S3 v1 L4 K& |. B# t2 R  [4 G1 X; L
  9.      );
    6 I8 \0 `+ e& B, e* w/ u& }; V
  10. }
复制代码
这样,您就可以实际部署 SBT item collection。此外,您还可以创建一个独立的没有collection SBT item。我们不会掌握这些item的编程,我只是想向大家展示它在更大范围内的工作原理。
4 Z0 ^. E- s, a* y; V# Z
结论
% J4 R! U6 L$ t1 I/ m0 X. M
从第 3 章中非常简单的逻辑开始,我们已经了解了 NFT 或 Jettons 等复杂合约中的许多内容。这就是一个很好的例子,你可以一步一步地学习某些语法和概念,然后就可以学习更复杂的逻辑。希望这几节课能帮助你理解 Jettons 和 NFTs 的真正含义,以及如何处理它们的代码库--弄清楚它们是如何工作的,并在使用 TON 构建时学习更多可用的语法和架构。
非常感谢您的关注。对你们来说,这可能是最难的一章,虽然我们没有编写任何代码;我们只是在阅读一些你们从未见过的东西,而且非常复杂。你们能坚持到最后一课,我真的很骄傲,我期待着在接下来的章节中看到你们。希望你们喜欢目前的课程。
9 q' W' ~: e3 y/ ?+ v! ?9 x
' {$ C: e1 ~+ V7 J7 m5 Z
2 t- ^  E% h! ]! o. g* |
* T. T5 ?! J; N) N, n7 m6 B0 t
8 F9 M$ }7 h8 i3 U6 ]
/ D2 d  I. K2 F. G5 V
7 E8 n$ K- Q$ j. Q" ?
  Y, _" K, Q) o/ Q' @1 e+ T
分享到:
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则