在本课中,我们将回顾经典的 NFT 项目的代码。此外,我们还将学习 SoulBound (SBT) Item。
7 c6 A5 c X% p+ O+ f7 w) PNFT Item合约- int min_tons_for_storage() asm "50000000 PUSHINT"; ;; 0.05 TON
& |2 U2 A, x* n/ H
复制代码就像在前面的课程中一样,我们看到有一些常量显示了我们应该在合约上存储的最低代币数量,以便能够支付租金。 - n2 J9 H# z6 \# Z% ^
- (int, int, slice, slice, cell) load_data() {
* v+ R$ s1 [4 d0 I S+ z - slice ds = get_data().begin_parse();% S( K% @5 o( C v7 s; v: p @
- var (index, collection_address) = (ds~load_uint(64), ds~load_msg_addr());
+ h& F4 f0 Q& L, P0 R3 N: d1 [ - if (ds.slice_bits() > 0) {; I/ @. K4 C: C1 F
- return (-1, index, collection_address, ds~load_msg_addr(), ds~load_ref());
; N9 S0 A% N% | - } else { " s" ]" g, T. o1 Z0 `! ]
- return (0, index, collection_address, null(), null()); ;; nft not initialized yet% f, S8 I3 I/ a0 M% [
- }
# y( Q2 F5 v, a - }
复制代码然后我们看到 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 {) e" [. |+ ]! p% i2 o
- set_data(8 l, D$ H$ e3 r7 L
- begin_cell()/ J7 E6 [# v4 [+ n* U
- .store_uint(index, 64)
# x$ }5 u# u! w - .store_slice(collection_address)
8 d: b& l6 T0 E& ^# G, r - .store_slice(owner_address)4 ?0 C7 s4 f6 P( s$ U, y5 d/ _
- .store_ref(content)
1 w# O, g/ [+ `6 V; O- p+ g% H X - .end_cell()
$ ?) }4 {$ q; {1 ]5 C/ t" o - );
( Y; u$ J( W4 m - }
复制代码我们还看到 store_data 就像我们调查的其他合约一样,这只是正常情况。
, |( f. r& }' s& a3 h- () 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 - var msg = begin_cell()
L0 n! } w* { - .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 - .store_slice(to_address). D& Y* t8 J, y
- .store_coins(amount); Y: P/ x2 p+ x& Q6 R
- .store_uint(0, 1 + 4 + 4 + 64 + 32 + 1 + 1)
5 u8 D) j* Q* m5 P2 w/ w - .store_uint(op, 32)9 \/ G) s' K n
- .store_uint(query_id, 64);) S _6 [. D3 h4 p& \1 F0 s' h
/ {8 m8 y+ S' }9 z# p/ F- if (~ builder_null?(payload)) { U; ` N0 [. O0 ?' b" @: X
- msg = msg.store_builder(payload);
3 r9 ~# D! Z7 l+ n% b - }
复制代码在这里,我们封装了发送消息的逻辑,因此我们可以在合约的其他逻辑中使用这个函数。因此,只要我们想发送信息,就可以使用这个函数。它接受目标地址、我们应附加的金额、操作代码、查询 ID、可能的有效载荷(这次是以生成器格式)以及消息的发送模式。这就是我们发送信息的逻辑。 我们的另一个函数是 transfer_ownership. 让我们稍后再研究,一旦我们在 recv_internal 逻辑,因为它是这里最重要的功能。 - () recv_internal(int my_balance, int msg_value, cell in_msg_full, slice in_msg_body) impure {
% Z1 s: ~) o& `& g - if (in_msg_body.slice_empty?()) { ;; ignore empty messages' x% E. `' N7 ^( J
- return ();& d \) ~" l: z
- }$ G' S; Y% B9 w8 r4 u) } k; X# u
1 I8 h2 p3 f& N3 Z( e! A- slice cs = in_msg_full.begin_parse();- p, ?" q8 _2 J/ j7 F& B6 c/ E
- int flags = cs~load_uint(4);2 Z- {' K0 o) j
- ; D5 [) d) }: ?$ D$ k
- if (flags & 1) { ;; ignore all bounced messages
( W0 g+ ]' j7 C. }, w - return ();
% K/ I5 r, e# @) c/ ]5 t - }4 {7 s% V& y9 u' a) J
- slice sender_address = cs~load_msg_addr();
+ H& _ A+ }" K% R
7 T/ O- C) _9 x8 E# ?* d! r- cs~load_msg_addr(); ;; skip dst( P# z- z0 y+ r+ o9 J: \4 ^& N, S
- cs~load_coins(); ;; skip value
+ ?; Y# k( r. X, e4 G, ? - cs~skip_bits(1); ;; skip extracurrency collection4 r& m2 C4 k& T8 }) i# p
- cs~load_coins(); ;; skip ihr_fee. ~3 j" b- H5 }
- 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! ?
P! B5 |& l l0 A- (int init?, int index, slice collection_address, slice owner_address, cell content) = load_data();
0 e% R6 b7 c0 ^# j- P9 V: V - if (~ init?) {. R/ ~" l8 @3 Z: o. C5 A4 r) `$ P8 D) E
- throw_unless(405, equal_slices(collection_address, sender_address));
0 Z# u& N; t- w1 \5 L: g - 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
- return ();; D; g* d0 e; p4 T9 ~ M7 G, o
- }
复制代码我们来看看它是怎么做的。我们再次忽略空的 in_msg_body 消息,读取标志,忽略退回的信息,并检查发件人是谁。现在我们跳过一些内容,比如信息的目标地址之类的,因为你已经知道这个Cell的内容是什么了。接下来是 fwd_fee 再次根据上一笔交易估算发送下一笔交易的成本。 然后加载数据。正如你所记得的,我们加载数据的方式与这里的实现方式相同。因此,我们要检查它是否未初始化,然后我们期待实际获得这些数据,期待 in_msg_body 以获得所有者的地址和带有元数据的Cell。但这只能通过 collection_address, 因此我们要检查它是否等于发件人地址。然后我们就可以初始化 NFT Item了。 - int op = in_msg_body~load_uint(32);' N) I5 q3 w4 ~' @% w
- int query_id = in_msg_body~load_uint(64);
1 x6 d! @+ \8 e3 Y! w' B; A
- M+ F+ m* Z% c7 {1 C$ n5 g1 H# U- if (op == op::transfer()) {6 F2 e- [& r [( f+ e& v2 n
- transfer_ownership(my_balance, index, collection_address, owner_address, content, sender_address, query_id, in_msg_body, fwd_fee);
& l1 c$ ]" s3 R - return ();
: J! d4 D7 ?: U5 r, u. r3 f* k; i - }
/ z" J, R* O: Q. F1 p - if (op == op::get_static_data()) {% }7 Y K$ A4 i( J. e1 K
- 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 - return ();
\4 B3 E% m- N8 E5 g% b3 { - }
复制代码接下来是操作代码,可以是 op::transfer 或 op::get_static_data. 这是我们要处理的两个可能值。当我们说 get_static_data, 我们会立即向发送静态数据请求的用户发送一条信息。我们只是报告数据,所以我们会向他发送 index 和 collection_address. 这是 get_static_data 函数。正如你所看到的,这是我们的第一个 send_ msg 用例 , {8 K5 ?( Y. x2 @ e2 Y" o6 ^
- () 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 - throw_unless(401, equal_slices(sender_address, owner_address));0 l9 ]! k+ N# K# q
- 0 I( d% ~6 k7 U' v. S5 _. C1 `
- slice new_owner_address = in_msg_body~load_msg_addr();
N. B( |( z+ I$ W h - force_chain(new_owner_address);( H2 y$ n) x$ ^4 `+ ~
- slice response_destination = in_msg_body~load_msg_addr();5 C& H' C( W4 n. C
- in_msg_body~load_int(1); ;; this nft don't use custom_payload
$ o5 F/ [7 {! T; Z- K - int forward_amount = in_msg_body~load_coins();
3 f# G! }% P- A I S a$ Y6 Q - throw_unless(708, slice_bits(in_msg_body) >= 1);: o _! K8 S, J+ d+ A+ R$ Y+ A
- ) W7 U% T9 @4 ?% K: G: n
- int rest_amount = my_balance - min_tons_for_storage();
! P9 U' S2 Y3 F. H$ r: v7 P - if (forward_amount) {- J& v" g6 ?7 f8 a% n+ W
- rest_amount -= (forward_amount + fwd_fees);
2 P- J" S' H V" y! f9 Y7 { - }4 _1 W( {% S5 t% b6 J. [
- int need_response = response_destination.preload_uint(2) != 0; ;; if NOT addr_none: 00
$ C; a* h2 ?$ A1 b& [( ^) ] - if (need_response) {
& k" p5 U5 s. i6 A - rest_amount -= fwd_fees;) z$ J) P& n0 C; |- T' p
- }
( I5 Z1 A. D( C
9 Z6 Z% }9 c7 p- 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# ^
1 J( y& a4 o. O0 x8 k; I0 Q- if (forward_amount) {
$ E: j$ N K% v+ q; ?( h1 Q: i* ~. j - 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
- }+ x; Y; K# L) \' N: G! D8 u, ^
- if (need_response) {4 ?) R6 e- `- i/ K) B$ o
- force_chain(response_destination);+ K+ W" }4 e7 Q+ D0 v
- send_msg(response_destination, rest_amount, op::excesses(), query_id, null(), 1); ;; paying fees, revert on errors% H9 k" h7 y: l, C
- }
' f3 h% X O* P/ X
4 t/ q: Q0 K3 f* _* J* d( f- store_data(index, collection_address, new_owner_address, content);; ]* D' q: Z7 M) K) K6 O) X5 r
- }
复制代码这里有一个转移所有权的功能。首先,我们要检查物品的所有者地址是否就是发送信息的人,因为只有他才能转移所有权。然后,我们读出新所有者的地址。我们 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合约GetGems repository 正在主办一些不同的 NFT 合约、藏品、销售和市场以及一些拍卖活动。这里是了解更多与 NFT 藏品、物品及其相关的合约的好地方。但我想向您展示一份确切的合约,我们可以对其进行深入研究。 sbt-item 是一种灵魂绑定代币(SBT)。灵魂绑定与 NFT 非常相似,但不同之处在于灵魂绑定代币不能擅自转让给其他人。我们仔细看看它的代码。 - global int storage::index;
) y9 M7 }% c) D* `/ c; `1 ]% Q - global int init?;
" c6 Q* S8 ?9 Q - global slice storage::collection_address;
7 w: U# }' d+ J6 T l( {. W - global slice storage::owner_address; l3 P3 W1 W4 ^# j- e) H
- global slice storage::authority_address;2 E% E" C: [. o
- global cell storage::content;1 W- ^1 g3 H4 Y0 B! f2 d+ a
- global int storage::revoked_at;7 T' g* D. ^+ t+ b T
- () load_data() impure {# j& K; Q: T5 l1 U
- slice ds = get_data().begin_parse();
1 D/ V/ g, t# }. ` - ( U) z1 c3 I" b4 P$ H, V; N& z
- storage::index = ds~load_uint(64);
& x6 d- {# _1 V# p8 U7 a - storage::collection_address = ds~load_msg_addr();
% V: G5 l- i4 l - init? = false;
" Z/ ?( L: Q; S# Q/ Z - / x* u2 R5 T% {3 @: Y9 O+ H
- if (ds.slice_bits() > 0) {9 M& O1 |+ m7 V: s9 Q! F
- init? = true;! N" V* @+ Y7 V. r4 r& @! Z
- storage::owner_address = ds~load_msg_addr();! ^) R9 \+ ~7 k9 j- m
- storage::content = ds~load_ref();
) X, N% `+ F; d* B) Q - storage::authority_address = ds~load_msg_addr();
) E! ~5 y$ ~, Q/ g+ J$ M! N# K - storage::revoked_at = ds~load_uint(64);/ h% T1 D( k: ` [: @7 P& h& b
- }
( b# ~/ P9 ^) }6 X$ k& k - }
复制代码这里有一些变量: storage::index, storage::collection_address. 我们还定义了其他全局变量,这只是 GetGems 实现的一个例子。你不必用同样的方法。我想在这里重点谈谈其他部分:
/ m L+ J% r, y6 Z# o3 F* |& _- if (op == op::request_owner()) {" l( n8 V- g7 @0 N2 d. g! L
- slice dest = in_msg_body~load_msg_addr();
" Z/ @! d* G+ o$ p3 ~: s! G) m# l - cell body = in_msg_body~load_ref();7 O7 [( A- I, x% Y7 }$ N
- int with_content = in_msg_body~load_uint(1);9 ^' w' X8 U# C8 J1 |
" _2 a; d: S+ m- var msg = begin_cell()
0 v5 P- ~+ B0 R# _ - .store_uint(storage::index, 256)4 P: ^3 N$ w$ G9 J
- .store_slice(sender_address)
2 J9 X% G' N: t: B - .store_slice(storage::owner_address)0 R9 D' B) _4 l6 i
- .store_ref(body)
4 b7 n# D/ ?; K/ K! F - .store_uint(storage::revoked_at, 64)! |1 J9 C: i4 d0 E( n6 R
- .store_uint(with_content, 1);
( e, ?1 p4 x& [+ Z- q8 D2 e( ] - / W6 B- g& Q# n9 T
- if (with_content != 0) {& k$ s+ o1 K# s0 d) f: G' l
- msg = msg.store_ref(storage::content);
8 z" x# C" l2 k' n# }% D. a; e% s - }( S1 S- \3 `$ Y5 M* m# f3 ~
4 i/ ~2 n; ~9 c! L8 W- ;; mode 64 = carry all the remaining value of the inbound message$ G% W: X; a2 p8 O5 s2 N& f2 t: d
- send_msg(flag::regular() | flag::bounce(), dest, 0, op::owner_info(), query_id, msg, 64);0 `4 C$ k- F W6 {# V& P' P
- return ();" d( [) S/ ]* {% |$ O( [
- }
! j5 J/ x7 n1 t a- ]9 x& H, v - if (op == op::prove_ownership()) {% }1 m" ]. R) R% t
- throw_unless(401, equal_slices(storage::owner_address, sender_address));1 y- G! z$ N* s% b" @: ~ }
+ ]( o w+ I( q! O2 Y- x- slice dest = in_msg_body~load_msg_addr();
" e9 K8 r. _& D0 W, K( j - cell body = in_msg_body~load_ref();7 t5 ?: f& \9 s3 A
- int with_content = in_msg_body~load_uint(1);
+ ^9 B; F2 l) u( K( s - 6 O$ Z+ @$ R& [! W7 t/ @
- var msg = begin_cell(): P8 T1 g0 b% a9 _) q M$ X
- .store_uint(storage::index, 256)
3 e& m) J6 H) `1 S* \2 C - .store_slice(storage::owner_address); e- w5 L# N: U
- .store_ref(body)
: }" e2 k7 \' O5 z; z3 ?- T& ] - .store_uint(storage::revoked_at, 64), t/ _8 {3 a; J) Z6 K! ?$ ~
- .store_uint(with_content, 1);/ X3 @0 l' D9 C1 x' M
$ u' H+ e% e [$ h; U3 t+ \- if (with_content != 0) {) n/ t" r1 r2 L
- msg = msg.store_ref(storage::content);
) U& r5 L2 ^. h* j- p6 I8 J - }& J5 J4 i& W1 s
- * a/ q' n0 ]- j' S
- ;; mode 64 = carry all the remaining value of the inbound message
0 k4 |/ D# C8 O3 X6 L+ A - send_msg(flag::regular() | flag::bounce(), dest, 0, op::ownership_proof(), query_id, msg, 64);
4 _0 p1 x: n4 Z: e4 k - return ();% n; z/ k) f" p# I/ Q" {0 f
- }
/ X# t' H: V e. n - if (op == op::get_static_data()) {3 l8 e# ]5 _* J% F
- var msg = begin_cell().store_uint(storage::index, 256).store_slice(storage::collection_address);0 V' j; b8 u1 ?' C
) C/ s7 [; @$ o) o- ;; mode 64 = carry all the remaining value of the inbound message5 d7 P& e& N |% U) m7 L8 ?( M$ T# q
- send_msg(flag::regular(), sender_address, 0, op::report_static_data(), query_id, msg, 64);
; w# u* I% M" ~# B - return ();* I% z A5 K: O
- }
$ R+ l4 g, @- F- a3 P' d - if (op == op::destroy()) {
* ? X7 @; x$ R1 z3 c1 j; G - throw_unless(401, equal_slices(storage::owner_address, sender_address));$ \. h8 g6 }* E" _1 R" s4 J
5 O7 z' h; }: K' N( W- send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);
) `( [3 R) A" k+ |3 z7 ~$ l - ' m* i$ R( d' Z0 p+ e* @
- storage::owner_address = null_addr();5 I, I: B S5 T3 R6 F
- storage::authority_address = null_addr();+ F9 l8 A9 E- A1 t" o |8 _
- store_data();1 Q5 q0 t" f* V& l5 T% ]# x: ]
- return ();( ?# G( p w: X! R* v; z
- }* J5 z; ^3 \) G+ x+ N8 W
- if (op == op::revoke()) {
9 `1 f4 z# D9 t& E - throw_unless(401, equal_slices(storage::authority_address, sender_address));
/ W! n5 L! L% V - throw_unless(403, storage::revoked_at == 0);5 C1 |6 [3 ^8 K0 v
- , U$ X$ U9 L* T$ @! f
- storage::revoked_at = now();! O9 u% ~! h, x" U
- store_data();. T& I. P, z; l' K- ^$ s- |
- return ();
" A$ Q3 [' U$ I: K - }
& H& D* h. z1 @" i - if (op == op::take_excess()) {2 Q+ f1 F+ c: p
- throw_unless(401, equal_slices(storage::owner_address, sender_address)); p1 J& _! g( U% {
- ; }* D4 O+ f& |; u3 R3 b- A
- ;; reserve amount for storage
2 |9 T3 P. }+ ]+ y% K, x4 H" _+ } - raw_reserve(min_tons_for_storage(), 0);( p; Y) d3 c9 x+ z: @3 s, l3 K8 u6 z! |0 H
) I$ O9 P0 K2 C E' V4 V d- send_msg(flag::regular(), sender_address, 0, op::excesses(), query_id, null(), 128);; ~5 u/ t: @- C. @, c% l& l
- return ();' A) l4 S3 M* s3 ?( Z5 ?+ P
- }
! M: e) q/ G* b7 R - if (op == op::transfer()) {& [( R" |( g& h: e3 z: o, g; J& r
- throw(413);
* ~1 U% C+ g* ~9 u3 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 {
6 G% B5 D% {0 ^ - var ds = get_data().begin_parse();
. T5 D7 ]" E F0 f; W | - return
# `& c- C; \. r$ K' Z - (ds~load_msg_addr(), ;; owner_address8 z: i& z. z" ?$ O. n" C |
- ds~load_uint(64), ;; next_item_index
. F! C5 o3 U |# a$ ] - ds~load_ref(), ;; content
4 O: o y2 z3 w1 c( F% Z9 { - ds~load_ref(), ;; nft_item_code
3 Y$ c/ T K% e4 a7 O. { - ds~load_ref() ;; royalty_params
5 l% S3 v1 L4 K& |. B# t2 R [4 G1 X; L - );
6 I8 \0 `+ e& B, e* w/ u& }; V - }
复制代码这样,您就可以实际部署 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
|