1use std::marker::PhantomData;
2
3use cosmwasm_schema::cw_serde;
4use cosmwasm_std::{
5 to_binary, Addr, CosmosMsg, CustomMsg, QuerierWrapper, StdResult, WasmMsg, WasmQuery,
6};
7use bs721::{
8 AllNftInfoResponse, Approval, ApprovalResponse, ApprovalsResponse, ContractInfoResponse,
9 NftInfoResponse, NumTokensResponse, OperatorsResponse, OwnerOfResponse, TokensResponse,
10};
11use serde::de::DeserializeOwned;
12use serde::Serialize;
13
14use crate::{ExecuteMsg, QueryMsg};
15
16#[cw_serde]
17pub struct Bs721Contract<Q: CustomMsg, E: CustomMsg>(
18 pub Addr,
19 pub PhantomData<Q>,
20 pub PhantomData<E>,
21);
22
23#[allow(dead_code)]
24impl<Q: CustomMsg, E: CustomMsg> Bs721Contract<Q, E> {
25 pub fn addr(&self) -> Addr {
26 self.0.clone()
27 }
28
29 pub fn call<T: Serialize>(&self, msg: ExecuteMsg<T, E>) -> StdResult<CosmosMsg> {
30 let msg = to_binary(&msg)?;
31 Ok(WasmMsg::Execute {
32 contract_addr: self.addr().into(),
33 msg,
34 funds: vec![],
35 }
36 .into())
37 }
38
39 pub fn query<T: DeserializeOwned>(
40 &self,
41 querier: &QuerierWrapper,
42 req: QueryMsg<Q>,
43 ) -> StdResult<T> {
44 let query = WasmQuery::Smart {
45 contract_addr: self.addr().into(),
46 msg: to_binary(&req)?,
47 }
48 .into();
49 querier.query(&query)
50 }
51
52 pub fn owner_of<T: Into<String>>(
55 &self,
56 querier: &QuerierWrapper,
57 token_id: T,
58 include_expired: bool,
59 ) -> StdResult<OwnerOfResponse> {
60 let req = QueryMsg::OwnerOf {
61 token_id: token_id.into(),
62 include_expired: Some(include_expired),
63 };
64 self.query(querier, req)
65 }
66
67 pub fn approval<T: Into<String>>(
68 &self,
69 querier: &QuerierWrapper,
70 token_id: T,
71 spender: T,
72 include_expired: Option<bool>,
73 ) -> StdResult<ApprovalResponse> {
74 let req = QueryMsg::Approval {
75 token_id: token_id.into(),
76 spender: spender.into(),
77 include_expired,
78 };
79 let res: ApprovalResponse = self.query(querier, req)?;
80 Ok(res)
81 }
82
83 pub fn approvals<T: Into<String>>(
84 &self,
85 querier: &QuerierWrapper,
86 token_id: T,
87 include_expired: Option<bool>,
88 ) -> StdResult<ApprovalsResponse> {
89 let req = QueryMsg::Approvals {
90 token_id: token_id.into(),
91 include_expired,
92 };
93 let res: ApprovalsResponse = self.query(querier, req)?;
94 Ok(res)
95 }
96
97 pub fn all_operators<T: Into<String>>(
98 &self,
99 querier: &QuerierWrapper,
100 owner: T,
101 include_expired: bool,
102 start_after: Option<String>,
103 limit: Option<u32>,
104 ) -> StdResult<Vec<Approval>> {
105 let req = QueryMsg::AllOperators {
106 owner: owner.into(),
107 include_expired: Some(include_expired),
108 start_after,
109 limit,
110 };
111 let res: OperatorsResponse = self.query(querier, req)?;
112 Ok(res.operators)
113 }
114
115 pub fn num_tokens(&self, querier: &QuerierWrapper) -> StdResult<u64> {
116 let req = QueryMsg::NumTokens {};
117 let res: NumTokensResponse = self.query(querier, req)?;
118 Ok(res.count)
119 }
120
121 pub fn contract_info(&self, querier: &QuerierWrapper) -> StdResult<ContractInfoResponse> {
123 let req = QueryMsg::ContractInfo {};
124 self.query(querier, req)
125 }
126
127 pub fn nft_info<T: Into<String>, U: DeserializeOwned>(
129 &self,
130 querier: &QuerierWrapper,
131 token_id: T,
132 ) -> StdResult<NftInfoResponse<U>> {
133 let req = QueryMsg::NftInfo {
134 token_id: token_id.into(),
135 };
136 self.query(querier, req)
137 }
138
139 pub fn all_nft_info<T: Into<String>, U: DeserializeOwned>(
141 &self,
142 querier: &QuerierWrapper,
143 token_id: T,
144 include_expired: bool,
145 ) -> StdResult<AllNftInfoResponse<U>> {
146 let req = QueryMsg::AllNftInfo {
147 token_id: token_id.into(),
148 include_expired: Some(include_expired),
149 };
150 self.query(querier, req)
151 }
152
153 pub fn tokens<T: Into<String>>(
155 &self,
156 querier: &QuerierWrapper,
157 owner: T,
158 start_after: Option<String>,
159 limit: Option<u32>,
160 ) -> StdResult<TokensResponse> {
161 let req = QueryMsg::Tokens {
162 owner: owner.into(),
163 start_after,
164 limit,
165 };
166 self.query(querier, req)
167 }
168
169 pub fn all_tokens(
171 &self,
172 querier: &QuerierWrapper,
173 start_after: Option<String>,
174 limit: Option<u32>,
175 ) -> StdResult<TokensResponse> {
176 let req = QueryMsg::AllTokens { start_after, limit };
177 self.query(querier, req)
178 }
179
180 pub fn has_metadata(&self, querier: &QuerierWrapper) -> bool {
182 self.contract_info(querier).is_ok()
183 }
184
185 pub fn has_enumerable(&self, querier: &QuerierWrapper) -> bool {
187 self.tokens(querier, self.addr(), None, Some(1)).is_ok()
188 }
189}