在本课中,我们将回顾经典的 NFT 项目的代码。此外,我们还将学习 SoulBound (SBT) Item。 " M4 m; T e7 _4 n+ Z7 L' K k
NFT Item合约- int min_tons_for_storage() asm "50000000 PUSHINT"; ;; 0.05 TON* M5 w8 r9 _' p Y$ V2 j
复制代码就像在前面的课程中一样,我们看到有一些常量显示了我们应该在合约上存储的最低代币数量,以便能够支付租金。 7 o5 z" c3 E3 m7 A5 x
- (int, int, slice, slice, cell) load_data() {
# I, J: |: k, Z - slice ds = get_data().begin_parse();' K4 g: \. N4 M+ o3 z5 v4 v9 L
- var (index, collection_address) = (ds~load_uint(64), ds~load_msg_addr());' n9 v: |5 Z& C/ s# U# \1 K" u
- if (ds.slice_bits() > 0) {" i/ r- _' a& i S7 K
- return (-1, index, collection_address, ds~load_msg_addr(), ds~load_ref());
6 Z D7 ~$ N% b7 n$ z2 M - } else { + v; t% }" T% _3 _
- return (0, index, collection_address, null(), null()); ;; nft not initialized yet
( x. [ N3 e/ g - }$ b7 V3 H5 {# i# z) Z v3 j
- }
复制代码然后我们看到 load_data. 这个 load_data 与我们以前看到的有些不同。通常,我们只是从本地存储区读取数据,但在这里,我们有一些更复杂的逻辑。因此,我们打开本地存储的Cell进行读取,然后读出 index 和 collection_address. 我们知道什么是索引和 Collections 地址。 然后,我们尝试读取 NFT Item的元数据。首先,我们检查它是否存在。如果是,我们返回 –1, 以及Collections地址、所有者地址和元数据。或者我们只返回 0, 索引和Collections地址时,没有所有者和元数据。这意味着 NFT 尚未初始化。基于这一点,我们将在后面的代码中做出一些决定。 - () store_data(int index, slice collection_address, slice owner_address, cell content) impure {( b* f: k: ?6 O/ S; n
- set_data(2 z/ E5 T) Y+ c
- begin_cell()
* s5 N: n% w% E5 u+ ^& `% W - .store_uint(index, 64); p- B O, c/ j, x0 a5 X& ?, M
- .store_slice(collection_address)4 U# F" x) A7 l& q! R
- .store_slice(owner_address)
3 n" S J! b/ c# i/ f - .store_ref(content)
# d6 }: V# w( D( L2 y7 r& [- ]& u+ b - .end_cell()* B1 |: }/ a+ J9 }
- );& m. }# R, f, h* w$ o- y
- }
复制代码我们还看到 store_data 就像我们调查的其他合约一样,这只是正常情况。
( |# _# U% z/ h! T- () send_msg(slice to_address, int amount, int op, int query_id, builder payload, int send_mode) impure inline {
; k2 c- @- @3 |5 ] - var msg = begin_cell()- B) D9 l; K! ] c' ^; }
- .store_uint(0x10, 6) ;; nobounce - int_msg_info$0 ihr_disabled:Bool bounce:Bool bounced:Bool src:MsgAddress -> 010000
" U1 @! u% i2 z: l. G - .store_slice(to_address), z" \" G7 F* U( F+ N
- .store_coins(amount)7 [& S+ u# I0 D
- .store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1)1 F0 |* w( e3 ]1 P9 z3 W, a! e
- .store_uint(op, 32)
P ?( q0 ?- ?$ {: T3 | - .store_uint(query_id, 64);- l. L# U6 ?$ b7 V; c# ^
- " }; ]! Y" \( }2 U
- if (~ builder_null?(payload)) {
- P8 A2 T- o6 R( ? - msg = msg.store_builder(payload);# Q5 G7 k7 e* b# K# ]
- }
复制代码在这里,我们封装了发送消息的逻辑,因此我们可以在合约的其他逻辑中使用这个函数。因此,只要我们想发送信息,就可以使用这个函数。它接受目标地址、我们应附加的金额、操作代码、查询 ID、可能的有效载荷(这次是以生成器格式)以及消息的发送模式。这就是我们发送信息的逻辑。 我们的另一个函数是 transfer_ownership. 让我们稍后再研究,一旦我们在 recv_internal 逻辑,因为它是这里最重要的功能。 - () recv_internal(int my_balance, int msg_value, cell in_msg_full, slice in_msg_body) impure {
. ?, A) `' ~, O$ Z7 h8 [' K9 n - if (in_msg_body.slice_empty?()) { ;; ignore empty messages
9 B0 V7 G$ G5 O' \ o$ A - return ();
% o2 q& p. M3 d q# T - }
3 L9 J5 ^4 g4 G0 R! H8 a4 n1 z
( g( p( t- V! [0 P0 V i$ J- slice cs = in_msg_full.begin_parse();
`8 [- O. \0 @ - int flags = cs~load_uint(4);. a+ c; C7 t1 l
: a+ p1 ^7 s! ^* B- v" U- if (flags & 1) { ;; ignore all bounced messages: s" M u- j' J+ }& i" h: F; {
- return ();
0 T* ]: u9 o3 q/ K1 f, s8 R - }$ O* @$ i. h& t2 r7 N$ T
- slice sender_address = cs~load_msg_addr();
. V5 ]) B, M( `8 u - ; I- j6 ]9 y# X
- cs~load_msg_addr(); ;; skip dst9 O' K4 T& d, K& ^. K: b4 Z
- cs~load_coins(); ;; skip value6 l$ { u0 Y) O9 ]$ ^9 @; ^
- cs~skip_bits(1); ;; skip extracurrency collection
& f1 H5 Y, l! f - cs~load_coins(); ;; skip ihr_fee& o: @6 ]- v8 g/ T# o
- int fwd_fee = muldiv(cs~load_coins(), 3, 2); ;; we use message fwd_fee for estimation of forward_payload costs# j$ Q @* j# a5 g
) V4 K) o4 D) q3 \7 D$ a n2 N! P- (int init?, int index, slice collection_address, slice owner_address, cell content) = load_data();6 f! q x+ J# E
- if (~ init?) {. M2 ^- Q: R& I& n
- throw_unless(405, equal_slices(collection_address, sender_address));
( M8 R% p! Q( w1 h6 Y7 h- X9 u - store_data(index, collection_address, in_msg_body~load_msg_addr(), in_msg_body~load_ref());% U) P, K. ^2 t) V8 W$ M1 e: M
- return ();
Z: x* d1 _4 g - }
复制代码我们来看看它是怎么做的。我们再次忽略空的 in_msg_body 消息,读取标志,忽略退回的信息,并检查发件人是谁。现在我们跳过一些内容,比如信息的目标地址之类的,因为你已经知道这个Cell的内容是什么了。接下来是 fwd_fee 再次根据上一笔交易估算发送下一笔交易的成本。 然后加载数据。正如你所记得的,我们加载数据的方式与这里的实现方式相同。因此,我们要检查它是否未初始化,然后我们期待实际获得这些数据,期待 in_msg_body 以获得所有者的地址和带有元数据的Cell。但这只能通过 collection_address, 因此我们要检查它是否等于发件人地址。然后我们就可以初始化 NFT Item了。 - int op = in_msg_body~load_uint(32);* ?( s+ Q7 F, ?4 o
- int query_id = in_msg_body~load_uint(64);
, M3 ^8 _/ p. A" C7 Q Y
. s; n$ o7 j k5 M- if (op == op::transfer()) {
) q1 g. L, F. q; Z4 f - transfer_ownership(my_balance, index, collection_address, owner_address, content, sender_address, query_id, in_msg_body, fwd_fee);
. Z4 h# u! M1 q& I1 F& x* I' S - return ();
k' U$ r4 q) f8 G w" W - }
- H4 e4 G; l+ Z - if (op == op::get_static_data()) {
4 R1 N) i- e) w" A/ } - 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
9 _2 E! v; l4 o& X - return ();& \9 ], }! o- Q3 [" Z
- }
复制代码接下来是操作代码,可以是 op::transfer 或 op::get_static_data. 这是我们要处理的两个可能值。当我们说 get_static_data, 我们会立即向发送静态数据请求的用户发送一条信息。我们只是报告数据,所以我们会向他发送 index 和 collection_address. 这是 get_static_data 函数。正如你所看到的,这是我们的第一个 send_ msg 用例 1 L i4 k% J- F; d5 b$ l& x5 e$ q
- () 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 {
D5 ~$ w C' Q/ N+ G - throw_unless(401, equal_slices(sender_address, owner_address));
# J ?" j$ v/ D5 v6 k) W3 N3 f( ~0 C- U - & C' x( c4 v7 x- v. Z: g
- slice new_owner_address = in_msg_body~load_msg_addr();
! r. H: q# Z* ?0 S0 b- Z - force_chain(new_owner_address);5 ~' T+ g v8 [2 U9 r# N
- slice response_destination = in_msg_body~load_msg_addr();
1 T. U+ I3 O; Z0 J1 [8 h) w - in_msg_body~load_int(1); ;; this nft don't use custom_payload7 G0 k2 F8 b0 Y8 x9 O
- int forward_amount = in_msg_body~load_coins();% |$ Y, x" _$ Q6 t E& x- V; [
- throw_unless(708, slice_bits(in_msg_body) >= 1);
; d7 Z. y& x; X5 Q$ n
# h: b N) |$ G/ Q- int rest_amount = my_balance - min_tons_for_storage();
5 z: ^) C" r. q* o - if (forward_amount) {5 `3 l" j! A1 R S0 w; k# R
- rest_amount -= (forward_amount + fwd_fees);- l& @! |3 @& Y3 R. y) ^
- }
M& C0 X: j" e* Y. Z% G - int need_response = response_destination.preload_uint(2) != 0; ;; if NOT addr_none: 00
/ w" G* n! E5 E7 k1 x - if (need_response) {: p# \( |4 w R; a9 { f: j+ L6 w% \
- rest_amount -= fwd_fees;
4 _! r! Y6 n& ?% }( T! o' F - }1 }# k& z- R! I7 \3 |) A
- s6 P1 z3 z" b
- throw_unless(402, rest_amount >= 0); ;; base nft spends fixed amount of gas, will not check for response
5 Q8 H q1 ]9 S2 Y1 E4 y - 2 G2 G8 i9 ^$ s6 N
- if (forward_amount) {+ c7 K8 f3 ~8 _- p
- 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
1 i. u+ I' f N+ S - }* {( q9 h' Y$ K; a% s- b" C7 f3 v+ c
- if (need_response) {# {1 t. @( C% R: U( `
- force_chain(response_destination);7 L: h e) d6 Z- O$ j8 D0 W
- send_msg(response_destination, rest_amount, op::excesses(), query_id, null(), 1); ;; paying fees, revert on errors* f$ v: i0 d4 `' {" _' {
- }' L: [* i! E+ x5 d; x
- , T1 O( q, S' I. q/ i1 `
- store_data(index, collection_address, new_owner_address, content);% D4 X$ z, }; R! b
- }
复制代码这里有一个转移所有权的功能。首先,我们要检查物品的所有者地址是否就是发送信息的人,因为只有他才能转移所有权。然后,我们读出新所有者的地址。我们 force_chain 以确保他在同一条链上。然后我们检查是否有 response_destination, 因此,一旦我们转移了这个所有权,我们是否应该告诉某人向另一个人发送信息。我们检查 forward_amount 是多少,如果有人要求的话。然后,我们就可以计算出,在发出信息之后,我们就会有足够的钱 forward_amount 到目的地。然后,我们要弄清楚是否需要回应,如果存在 response_destination 。如果存在,其剩余金额必须大于 0。 下面我就向大家简单介绍一下它的工作原理。如果存在 forward_amount ,我们将发送一条信息,通知用户所有权已由该 forward_amount 分配。如果我们需要回应,我们将 force_chain 以确保响应目的地在同一链上。我们发送的回复通常包括超额费用部分和所有额外资金;它们只是被转发到 response_destination. 然后,我们只需使用 new_owner_address 保存数据。基本上,这是唯一会发生变化的地方、 transfer_ownership 是这里的核心函数之一。您拥有一件物品,有时您会将其所有权转让给其他人或出售,这取决于交易的类型。
9 h5 a# E" _+ \& Q$ {* i' x- t SBT item合约GetGems repository 正在主办一些不同的 NFT 合约、藏品、销售和市场以及一些拍卖活动。这里是了解更多与 NFT 藏品、物品及其相关的合约的好地方。但我想向您展示一份确切的合约,我们可以对其进行深入研究。 sbt-item 是一种灵魂绑定代币(SBT)。灵魂绑定与 NFT 非常相似,但不同之处在于灵魂绑定代币不能擅自转让给其他人。我们仔细看看它的代码。 - global int storage::index;) [) c# k- D7 O b
- global int init?;# {; I3 P* g4 {* e2 O4 i
- global slice storage::collection_address;
/ [+ }( ?( ]& A5 Q" k - global slice storage::owner_address;
, |7 V( c, ^2 B# r# l - global slice storage::authority_address;9 o! Z, e* O6 ^
- global cell storage::content;
7 z2 l2 s; t c, C# f% } - global int storage::revoked_at;
! f! L V# B) }6 b - () load_data() impure {$ p4 `% \: a, O" p$ |9 r V3 Z
- slice ds = get_data().begin_parse();
% S/ H. x+ c) L+ p) |( w N+ t - L0 |2 x2 |+ H7 ?+ n v1 ]
- storage::index = ds~load_uint(64);+ I. A2 k) O! `& F3 j$ G
- storage::collection_address = ds~load_msg_addr();
. Y/ c9 @, b L! `% s% `8 j - init? = false;$ Z. s5 O) w t9 |& n( c, t5 p: M
! W$ F' }8 j% M" e0 x! ?$ P- if (ds.slice_bits() > 0) {
p b, B. B7 u - init? = true;
0 a, O6 I$ n' {+ f1 ~6 K: Y - storage::owner_address = ds~load_msg_addr();
! p' S9 R& K) P1 i7 e - storage::content = ds~load_ref();3 p/ h) d7 W7 ~
- storage::authority_address = ds~load_msg_addr();( G% g6 M6 p1 E, \& K8 y. c
- storage::revoked_at = ds~load_uint(64);6 r! h2 l7 V6 d( I3 e* V% C
- }
/ B- N/ J- q7 a+ V* K - }
复制代码这里有一些变量: storage::index, storage::collection_address. 我们还定义了其他全局变量,这只是 GetGems 实现的一个例子。你不必用同样的方法。我想在这里重点谈谈其他部分:
: d0 M0 o u Q4 F, f- if (op == op::request_owner()) {
7 k- a1 M/ G1 ]: t - slice dest = in_msg_body~load_msg_addr();# ]7 ]3 H& S' `9 G4 U
- cell body = in_msg_body~load_ref();$ @5 H/ @5 ~% s; M, j5 E; d/ {
- int with_content = in_msg_body~load_uint(1);
+ }1 N+ _6 J7 i H - 0 E' a! u- O$ q3 ?; s7 R
- var msg = begin_cell()# ?3 T! T# U7 J
- .store_uint(storage::index, 256)
$ \. ~8 c `4 X" g' {: o - .store_slice(sender_address)
5 i/ J4 L- O* i8 T3 t T - .store_slice(storage::owner_address)
- b9 f# g, C# H( M5 _4 r - .store_ref(body)3 u! D, ^: i/ O- Z9 D9 M
- .store_uint(storage::revoked_at, 64)
: H. f, d2 f0 [$ ^ - .store_uint(with_content, 1);
' [6 o5 _* _: f( r% C% ~: a/ L
% D- e, b1 _: F9 x6 J1 J, G3 g- v- if (with_content != 0) {1 p- w/ e$ p( `# }8 c6 O
- msg = msg.store_ref(storage::content);
; W5 p$ C7 M9 G - }
0 T; j. Y+ W- p6 R! G5 d5 m+ d
( u$ |7 k' l4 q$ T* C- ;; mode 64 = carry all the remaining value of the inbound message1 c3 @/ z2 j) @" N: @
- send_msg(flag::regular() | flag::bounce(), dest, 0, op::owner_info(), query_id, msg, 64);
- p, E/ v+ A# N9 r0 o - return ();- T, h7 v2 c- F, |
- }% w( S' k0 W( v; W% }% P) _
- if (op == op::prove_ownership()) {
9 @4 _! L+ @; c& W3 s7 } - throw_unless(401, equal_slices(storage::owner_address, sender_address));& J; \/ e# j. u6 f) Z
- ) x. G# s" `7 f' q, X
- slice dest = in_msg_body~load_msg_addr(); `7 h( b. l8 j" Z; g
- cell body = in_msg_body~load_ref();
0 g4 z8 E* f' U6 k, o - int with_content = in_msg_body~load_uint(1);( T5 @0 i/ e8 M" u3 @( {( n6 Y
- 8 v7 q$ i0 ~. j c) a
- var msg = begin_cell()& p8 q4 M- i2 ^5 H& o4 m# V
- .store_uint(storage::index, 256)
4 P# H% {3 j; h- T - .store_slice(storage::owner_address)' ?, `. c8 U1 Z3 H5 y) s7 f% J
- .store_ref(body)+ o6 }4 \" Q" P: @
- .store_uint(storage::revoked_at, 64)
( \ R( i W2 k4 R - .store_uint(with_content, 1);8 z% O' S! p' L3 h* p" @
4 b( s, i. M- h: i- if (with_content != 0) {2 d! B; _( S$ p% h+ G9 T) P
- msg = msg.store_ref(storage::content);0 p1 g8 o& u5 L- t5 A
- }3 c) k& L+ g3 k1 T7 i* U
- 7 e! m9 L5 J' Y9 |. |' R' V
- ;; mode 64 = carry all the remaining value of the inbound message. M3 d. ]2 G0 G- Y, T0 O2 J6 B0 @
- send_msg(flag::regular() | flag::bounce(), dest, 0, op::ownership_proof(), query_id, msg, 64);- B/ H3 ?: J. K ]. C+ d$ I
- return ();
- Q/ S& s0 I6 Q- M1 e; r# c - }. e: J" F" J* Q" B2 ]: T
- if (op == op::get_static_data()) {
* K& n; M- i1 Z' c- l6 I - var msg = begin_cell().store_uint(storage::index, 256).store_slice(storage::collection_address);/ j; R9 H. b$ ]8 s
- ) s: H) E' B7 S' s
- ;; mode 64 = carry all the remaining value of the inbound message: K/ ?3 L6 {% Q4 o
- send_msg(flag::regular(), sender_address, 0, op::report_static_data(), query_id, msg, 64);
4 q+ h- b' Y Y# p - return ();
7 v& n* e! n* s - }
" _: n' O' T Z; f - if (op == op::destroy()) {, I& A3 m( o: }
- throw_unless(401, equal_slices(storage::owner_address, sender_address));- x* g! ]* G+ @, E; a. d4 L0 K) V
- + t, {/ o; }1 k3 a+ a% ?
- send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);
5 k3 b. q+ C* R7 Q - 2 x$ }& S) H! I) I
- storage::owner_address = null_addr();4 ?2 G$ E7 k( y( X7 i" f2 p6 j) K
- storage::authority_address = null_addr();1 h7 c4 Q! d" M' ?8 z. n
- store_data();
- f# @, k9 w: @# n% q ^! p - return ();
+ Y) t+ M: K3 g/ Q - }- w( f# g: J9 e
- if (op == op::revoke()) {
5 F% L" H4 `" M3 ^ - throw_unless(401, equal_slices(storage::authority_address, sender_address));
5 R6 S: H7 Z8 N% w; } - throw_unless(403, storage::revoked_at == 0);$ i" H# v$ h0 P/ E
- 6 |3 t3 ?) @2 F5 j0 w
- storage::revoked_at = now();
$ T) G: J6 r' q+ Y# U - store_data();
h" m& u' K4 ]! D* p9 O# X. D2 b - return ();7 [3 E Q7 X: I6 S& i4 l
- }* Z9 u6 w6 r# m9 s- @3 U
- if (op == op::take_excess()) {
- c. {8 l5 }6 z/ N: G+ }0 k - throw_unless(401, equal_slices(storage::owner_address, sender_address));
9 W0 w$ N: t! C3 m. a
9 k) J1 B! ^! a8 D9 i; y- ;; reserve amount for storage
& y; C" d/ l1 [' o- L0 W% p/ m5 M - raw_reserve(min_tons_for_storage(), 0);
6 _% a* w, R7 V" _" H - " @# I' c! M% _0 ~# P/ s# k
- send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);
" H. s& h$ K3 O* c) U - return ();
2 N4 @0 \% \8 s" S( G) {( w - }
$ b O6 N7 N% E. r$ n8 M8 w& s - if (op == op::transfer()) {6 U9 ?5 Y5 n5 m$ v
- throw(413);8 n" p. D! S0 T" _+ z
- }
复制代码如你所见,这里有许多注释,但我们不会逐行深入研究。你可以看到,这个通证将被用于多种不同的用途,但与普通的 NFT 不同。这里是合约,你可以申请这个合约的所有者。一些操作代码可以请求所有权证明。您可以获取静态数据。你还可以销毁或撤销这个标记。你可以拿走多余的东西,比如,存放在这里的一些钱。如果你是这个合约的所有者,你就可以申请。但你显然不能转让它,就像你在这里看到的一样。这是一种完全不同类型的代币,如果你对 NFT 感兴趣,就应该在 GetGems 存储库里多花点时间。 我想向你展示的是,NFT collections 的所有代码可以保持不变,但collections合约部署的 item 可以不同。例如,你可以用 SBT item 替换 NFT item。这样您就会有一个 SBT item collection,然后您只需更改 code 来初始化你的 collection: - (slice, int, cell, cell, cell) load_data() inline {
8 s1 O' P( a" O' A - var ds = get_data().begin_parse();6 U8 o! d( E% T
- return
5 `$ b% S& l9 J/ Y1 ~" S - (ds~load_msg_addr(), ;; owner_address
" I# J1 Y* \# J" u; }2 [7 P) M - ds~load_uint(64), ;; next_item_index* G" m# L& t: a6 p/ t3 u
- ds~load_ref(), ;; content2 R. `2 \% d2 _
- ds~load_ref(), ;; nft_item_code
- ?- v: Y9 L: i. d: r! c4 {' }0 z a - ds~load_ref() ;; royalty_params
1 _( x" h7 ~5 C, C) D, X4 f: G+ @9 b - );
! v/ o+ k2 E/ {* ~/ m - }
复制代码这样,您就可以实际部署 SBT item collection。此外,您还可以创建一个独立的没有collection SBT item。我们不会掌握这些item的编程,我只是想向大家展示它在更大范围内的工作原理。 ; N% [4 B8 @+ m1 }0 l
结论! r2 A# S' n" X* G
从第 3 章中非常简单的逻辑开始,我们已经了解了 NFT 或 Jettons 等复杂合约中的许多内容。这就是一个很好的例子,你可以一步一步地学习某些语法和概念,然后就可以学习更复杂的逻辑。希望这几节课能帮助你理解 Jettons 和 NFTs 的真正含义,以及如何处理它们的代码库--弄清楚它们是如何工作的,并在使用 TON 构建时学习更多可用的语法和架构。 非常感谢您的关注。对你们来说,这可能是最难的一章,虽然我们没有编写任何代码;我们只是在阅读一些你们从未见过的东西,而且非常复杂。你们能坚持到最后一课,我真的很骄傲,我期待着在接下来的章节中看到你们。希望你们喜欢目前的课程。 ! ^" N6 i7 D" w& o) N0 s
: Q0 ~( r/ d3 J7 s* R
4 B# h. `' w5 P6 Q( V7 s
; Z& G/ z5 e0 s8 o
/ Q, M+ X2 W! h/ \& i D) |# ^- ` S) v( Z
$ v Z( d- Y3 ~; X) Z: S
: I+ C! Y2 L$ ~5 K1 e
|