ERC1155.sol 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. // SPDX-License-Identifier: MIT
  2. pragma solidity ^0.8.0;
  3. import "./IERC1155.sol";
  4. import "./IERC1155Receiver.sol";
  5. import "./extensions/IERC1155MetadataURI.sol";
  6. import "../../utils/Address.sol";
  7. import "../../utils/Context.sol";
  8. import "../../utils/introspection/ERC165.sol";
  9. /**
  10. * @dev Implementation of the basic standard multi-token.
  11. * See https://eips.ethereum.org/EIPS/eip-1155
  12. * Originally based on code by Enjin: https://github.com/enjin/erc-1155
  13. *
  14. * _Available since v3.1._
  15. */
  16. contract ERC1155 is Context, ERC165, IERC1155, IERC1155MetadataURI {
  17. using Address for address;
  18. // Mapping from token ID to account balances
  19. mapping (uint256 => mapping(address => uint256)) private _balances;
  20. // Mapping from account to operator approvals
  21. mapping (address => mapping(address => bool)) private _operatorApprovals;
  22. // Used as the URI for all token types by relying on ID substitution, e.g. https://token-cdn-domain/{id}.json
  23. string private _uri;
  24. /**
  25. * @dev See {_setURI}.
  26. */
  27. constructor (string memory uri_) {
  28. _setURI(uri_);
  29. }
  30. /**
  31. * @dev See {IERC165-supportsInterface}.
  32. */
  33. function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
  34. return interfaceId == type(IERC1155).interfaceId
  35. || interfaceId == type(IERC1155MetadataURI).interfaceId
  36. || super.supportsInterface(interfaceId);
  37. }
  38. /**
  39. * @dev See {IERC1155MetadataURI-uri}.
  40. *
  41. * This implementation returns the same URI for *all* token types. It relies
  42. * on the token type ID substitution mechanism
  43. * https://eips.ethereum.org/EIPS/eip-1155#metadata[defined in the EIP].
  44. *
  45. * Clients calling this function must replace the `\{id\}` substring with the
  46. * actual token type ID.
  47. */
  48. function uri(uint256) public view virtual override returns (string memory) {
  49. return _uri;
  50. }
  51. /**
  52. * @dev See {IERC1155-balanceOf}.
  53. *
  54. * Requirements:
  55. *
  56. * - `account` cannot be the zero address.
  57. */
  58. function balanceOf(address account, uint256 id) public view virtual override returns (uint256) {
  59. require(account != address(0), "ERC1155: balance query for the zero address");
  60. return _balances[id][account];
  61. }
  62. /**
  63. * @dev See {IERC1155-balanceOfBatch}.
  64. *
  65. * Requirements:
  66. *
  67. * - `accounts` and `ids` must have the same length.
  68. */
  69. function balanceOfBatch(
  70. address[] memory accounts,
  71. uint256[] memory ids
  72. )
  73. public
  74. view
  75. virtual
  76. override
  77. returns (uint256[] memory)
  78. {
  79. require(accounts.length == ids.length, "ERC1155: accounts and ids length mismatch");
  80. uint256[] memory batchBalances = new uint256[](accounts.length);
  81. for (uint256 i = 0; i < accounts.length; ++i) {
  82. batchBalances[i] = balanceOf(accounts[i], ids[i]);
  83. }
  84. return batchBalances;
  85. }
  86. /**
  87. * @dev See {IERC1155-setApprovalForAll}.
  88. */
  89. function setApprovalForAll(address operator, bool approved) public virtual override {
  90. require(_msgSender() != operator, "ERC1155: setting approval status for self");
  91. _operatorApprovals[_msgSender()][operator] = approved;
  92. emit ApprovalForAll(_msgSender(), operator, approved);
  93. }
  94. /**
  95. * @dev See {IERC1155-isApprovedForAll}.
  96. */
  97. function isApprovedForAll(address account, address operator) public view virtual override returns (bool) {
  98. return _operatorApprovals[account][operator];
  99. }
  100. /**
  101. * @dev See {IERC1155-safeTransferFrom}.
  102. */
  103. function safeTransferFrom(
  104. address from,
  105. address to,
  106. uint256 id,
  107. uint256 amount,
  108. bytes memory data
  109. )
  110. public
  111. virtual
  112. override
  113. {
  114. require(
  115. from == _msgSender() || isApprovedForAll(from, _msgSender()),
  116. "ERC1155: caller is not owner nor approved"
  117. );
  118. _safeTransferFrom(from, to, id, amount, data);
  119. }
  120. /**
  121. * @dev See {IERC1155-safeBatchTransferFrom}.
  122. */
  123. function safeBatchTransferFrom(
  124. address from,
  125. address to,
  126. uint256[] memory ids,
  127. uint256[] memory amounts,
  128. bytes memory data
  129. )
  130. public
  131. virtual
  132. override
  133. {
  134. require(
  135. from == _msgSender() || isApprovedForAll(from, _msgSender()),
  136. "ERC1155: transfer caller is not owner nor approved"
  137. );
  138. _safeBatchTransferFrom(from, to, ids, amounts, data);
  139. }
  140. /**
  141. * @dev Transfers `amount` tokens of token type `id` from `from` to `to`.
  142. *
  143. * Emits a {TransferSingle} event.
  144. *
  145. * Requirements:
  146. *
  147. * - `to` cannot be the zero address.
  148. * - `from` must have a balance of tokens of type `id` of at least `amount`.
  149. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
  150. * acceptance magic value.
  151. */
  152. function _safeTransferFrom(
  153. address from,
  154. address to,
  155. uint256 id,
  156. uint256 amount,
  157. bytes memory data
  158. )
  159. internal
  160. virtual
  161. {
  162. require(to != address(0), "ERC1155: transfer to the zero address");
  163. address operator = _msgSender();
  164. _beforeTokenTransfer(operator, from, to, _asSingletonArray(id), _asSingletonArray(amount), data);
  165. uint256 fromBalance = _balances[id][from];
  166. require(fromBalance >= amount, "ERC1155: insufficient balance for transfer");
  167. unchecked {
  168. _balances[id][from] = fromBalance - amount;
  169. }
  170. _balances[id][to] += amount;
  171. emit TransferSingle(operator, from, to, id, amount);
  172. _doSafeTransferAcceptanceCheck(operator, from, to, id, amount, data);
  173. }
  174. /**
  175. * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {_safeTransferFrom}.
  176. *
  177. * Emits a {TransferBatch} event.
  178. *
  179. * Requirements:
  180. *
  181. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
  182. * acceptance magic value.
  183. */
  184. function _safeBatchTransferFrom(
  185. address from,
  186. address to,
  187. uint256[] memory ids,
  188. uint256[] memory amounts,
  189. bytes memory data
  190. )
  191. internal
  192. virtual
  193. {
  194. require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
  195. require(to != address(0), "ERC1155: transfer to the zero address");
  196. address operator = _msgSender();
  197. _beforeTokenTransfer(operator, from, to, ids, amounts, data);
  198. for (uint256 i = 0; i < ids.length; ++i) {
  199. uint256 id = ids[i];
  200. uint256 amount = amounts[i];
  201. uint256 fromBalance = _balances[id][from];
  202. require(fromBalance >= amount, "ERC1155: insufficient balance for transfer");
  203. unchecked {
  204. _balances[id][from] = fromBalance - amount;
  205. }
  206. _balances[id][to] += amount;
  207. }
  208. emit TransferBatch(operator, from, to, ids, amounts);
  209. _doSafeBatchTransferAcceptanceCheck(operator, from, to, ids, amounts, data);
  210. }
  211. /**
  212. * @dev Sets a new URI for all token types, by relying on the token type ID
  213. * substitution mechanism
  214. * https://eips.ethereum.org/EIPS/eip-1155#metadata[defined in the EIP].
  215. *
  216. * By this mechanism, any occurrence of the `\{id\}` substring in either the
  217. * URI or any of the amounts in the JSON file at said URI will be replaced by
  218. * clients with the token type ID.
  219. *
  220. * For example, the `https://token-cdn-domain/\{id\}.json` URI would be
  221. * interpreted by clients as
  222. * `https://token-cdn-domain/000000000000000000000000000000000000000000000000000000000004cce0.json`
  223. * for token type ID 0x4cce0.
  224. *
  225. * See {uri}.
  226. *
  227. * Because these URIs cannot be meaningfully represented by the {URI} event,
  228. * this function emits no events.
  229. */
  230. function _setURI(string memory newuri) internal virtual {
  231. _uri = newuri;
  232. }
  233. /**
  234. * @dev Creates `amount` tokens of token type `id`, and assigns them to `account`.
  235. *
  236. * Emits a {TransferSingle} event.
  237. *
  238. * Requirements:
  239. *
  240. * - `account` cannot be the zero address.
  241. * - If `account` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155Received} and return the
  242. * acceptance magic value.
  243. */
  244. function _mint(address account, uint256 id, uint256 amount, bytes memory data) internal virtual {
  245. require(account != address(0), "ERC1155: mint to the zero address");
  246. address operator = _msgSender();
  247. _beforeTokenTransfer(operator, address(0), account, _asSingletonArray(id), _asSingletonArray(amount), data);
  248. _balances[id][account] += amount;
  249. emit TransferSingle(operator, address(0), account, id, amount);
  250. _doSafeTransferAcceptanceCheck(operator, address(0), account, id, amount, data);
  251. }
  252. /**
  253. * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {_mint}.
  254. *
  255. * Requirements:
  256. *
  257. * - `ids` and `amounts` must have the same length.
  258. * - If `to` refers to a smart contract, it must implement {IERC1155Receiver-onERC1155BatchReceived} and return the
  259. * acceptance magic value.
  260. */
  261. function _mintBatch(address to, uint256[] memory ids, uint256[] memory amounts, bytes memory data) internal virtual {
  262. require(to != address(0), "ERC1155: mint to the zero address");
  263. require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
  264. address operator = _msgSender();
  265. _beforeTokenTransfer(operator, address(0), to, ids, amounts, data);
  266. for (uint i = 0; i < ids.length; i++) {
  267. _balances[ids[i]][to] += amounts[i];
  268. }
  269. emit TransferBatch(operator, address(0), to, ids, amounts);
  270. _doSafeBatchTransferAcceptanceCheck(operator, address(0), to, ids, amounts, data);
  271. }
  272. /**
  273. * @dev Destroys `amount` tokens of token type `id` from `account`
  274. *
  275. * Requirements:
  276. *
  277. * - `account` cannot be the zero address.
  278. * - `account` must have at least `amount` tokens of token type `id`.
  279. */
  280. function _burn(address account, uint256 id, uint256 amount) internal virtual {
  281. require(account != address(0), "ERC1155: burn from the zero address");
  282. address operator = _msgSender();
  283. _beforeTokenTransfer(operator, account, address(0), _asSingletonArray(id), _asSingletonArray(amount), "");
  284. uint256 accountBalance = _balances[id][account];
  285. require(accountBalance >= amount, "ERC1155: burn amount exceeds balance");
  286. unchecked {
  287. _balances[id][account] = accountBalance - amount;
  288. }
  289. emit TransferSingle(operator, account, address(0), id, amount);
  290. }
  291. /**
  292. * @dev xref:ROOT:erc1155.adoc#batch-operations[Batched] version of {_burn}.
  293. *
  294. * Requirements:
  295. *
  296. * - `ids` and `amounts` must have the same length.
  297. */
  298. function _burnBatch(address account, uint256[] memory ids, uint256[] memory amounts) internal virtual {
  299. require(account != address(0), "ERC1155: burn from the zero address");
  300. require(ids.length == amounts.length, "ERC1155: ids and amounts length mismatch");
  301. address operator = _msgSender();
  302. _beforeTokenTransfer(operator, account, address(0), ids, amounts, "");
  303. for (uint i = 0; i < ids.length; i++) {
  304. uint256 id = ids[i];
  305. uint256 amount = amounts[i];
  306. uint256 accountBalance = _balances[id][account];
  307. require(accountBalance >= amount, "ERC1155: burn amount exceeds balance");
  308. unchecked {
  309. _balances[id][account] = accountBalance - amount;
  310. }
  311. }
  312. emit TransferBatch(operator, account, address(0), ids, amounts);
  313. }
  314. /**
  315. * @dev Hook that is called before any token transfer. This includes minting
  316. * and burning, as well as batched variants.
  317. *
  318. * The same hook is called on both single and batched variants. For single
  319. * transfers, the length of the `id` and `amount` arrays will be 1.
  320. *
  321. * Calling conditions (for each `id` and `amount` pair):
  322. *
  323. * - When `from` and `to` are both non-zero, `amount` of ``from``'s tokens
  324. * of token type `id` will be transferred to `to`.
  325. * - When `from` is zero, `amount` tokens of token type `id` will be minted
  326. * for `to`.
  327. * - when `to` is zero, `amount` of ``from``'s tokens of token type `id`
  328. * will be burned.
  329. * - `from` and `to` are never both zero.
  330. * - `ids` and `amounts` have the same, non-zero length.
  331. *
  332. * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
  333. */
  334. function _beforeTokenTransfer(
  335. address operator,
  336. address from,
  337. address to,
  338. uint256[] memory ids,
  339. uint256[] memory amounts,
  340. bytes memory data
  341. )
  342. internal
  343. virtual
  344. { }
  345. function _doSafeTransferAcceptanceCheck(
  346. address operator,
  347. address from,
  348. address to,
  349. uint256 id,
  350. uint256 amount,
  351. bytes memory data
  352. )
  353. private
  354. {
  355. if (to.isContract()) {
  356. try IERC1155Receiver(to).onERC1155Received(operator, from, id, amount, data) returns (bytes4 response) {
  357. if (response != IERC1155Receiver(to).onERC1155Received.selector) {
  358. revert("ERC1155: ERC1155Receiver rejected tokens");
  359. }
  360. } catch Error(string memory reason) {
  361. revert(reason);
  362. } catch {
  363. revert("ERC1155: transfer to non ERC1155Receiver implementer");
  364. }
  365. }
  366. }
  367. function _doSafeBatchTransferAcceptanceCheck(
  368. address operator,
  369. address from,
  370. address to,
  371. uint256[] memory ids,
  372. uint256[] memory amounts,
  373. bytes memory data
  374. )
  375. private
  376. {
  377. if (to.isContract()) {
  378. try IERC1155Receiver(to).onERC1155BatchReceived(operator, from, ids, amounts, data) returns (bytes4 response) {
  379. if (response != IERC1155Receiver(to).onERC1155BatchReceived.selector) {
  380. revert("ERC1155: ERC1155Receiver rejected tokens");
  381. }
  382. } catch Error(string memory reason) {
  383. revert(reason);
  384. } catch {
  385. revert("ERC1155: transfer to non ERC1155Receiver implementer");
  386. }
  387. }
  388. }
  389. function _asSingletonArray(uint256 element) private pure returns (uint256[] memory) {
  390. uint256[] memory array = new uint256[](1);
  391. array[0] = element;
  392. return array;
  393. }
  394. }