+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/_code-samples/mpt-generator/mpt-generator.js b/_code-samples/mpt-generator/mpt-generator.js
new file mode 100644
index 00000000000..082c523a68d
--- /dev/null
+++ b/_code-samples/mpt-generator/mpt-generator.js
@@ -0,0 +1,111 @@
+function generateCode() {
+ let v_flags = 0
+ if (clawbackSlider.checked) {v_flags+=64}
+ if (lockSlider.checked) {v_flags+=2}
+ if (authTokensSlider.checked) {v_flags +=4}
+ if (txrSlider.checked) {v_flags += 32}
+ if (tradeSlider.checked) {v_flags += 16}
+ if (escrowSlider.checked) {v_flags+=8}
+ const mptHexString = xrpl.convertStringToHex(metadataTextArea.value)
+ let v_codeBlock = "{\n \"TransactionType\": \"MPTokenIssuanceCreate\",\n \"Account\": \"" + accountField.value +
+ "\",\n \"AssetScale\": " + parseInt(assetScaleField.value) + ", \n \"MaximumAmount\": \"" + maximumAmountField.value +
+ "\",\n \"TransferFee\": " + transferFeeField.value +
+ ",\n \"Flags\": " + v_flags + ",\n \"MPTokenMetadata\": \"" + mptHexString + "\"\n}"
+ resultField.value = v_codeBlock
+} //End of generateCode()
+// ******************************************************
+// ************* Get the Preferred Network **************
+// ******************************************************
+function getNet() {
+ let net
+ if (document.getElementById("tn").checked) net = "wss://s.altnet.rippletest.net:51233"
+ if (document.getElementById("dn").checked) net = "wss://s.devnet.rippletest.net:51233"
+ return net
+} // End of getNet()
+
+// *******************************************************
+// ************* Get Account *****************************
+// *******************************************************
+
+async function getAccount() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ accountField.value = "Getting new account...."
+ let faucetHost = null
+ await client.connect()
+ let results = '\nConnected, funding wallet.'
+ resultField.value = results
+// ----------------------------------------Create and fund a test account wallet.
+ const my_wallet = (await client.fundWallet(null, { faucetHost })).wallet
+ results += '\nGot a wallet.'
+// ------------------------------------------------------Get the current balance.
+ const my_balance = (await client.getXrpBalance(my_wallet.address))
+ accountField.value = my_wallet.address
+ seedField.value = my_wallet.seed
+ results += '\nAccount created.'
+ resultField.value = results
+ client.disconnect()
+} // End of getAccount()
+
+// *******************************************************
+// ************ Get Account from Seed ********************
+// *******************************************************
+
+async function getAccountFromSeed() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ results = 'Connecting to ' + getNet() + '....'
+ await client.connect()
+ results += '\nConnected, finding wallets.\n'
+ resultField.value = results
+// --------------------------------------------------Find the account wallet.
+ const my_wallet = xrpl.Wallet.fromSeed(seedField.value)
+ accountField.value = my_wallet.address
+ seedField.value = my_wallet.seed
+ client.disconnect()
+} // End of getAccountFromSeed()
+
+ // *******************************************************
+ // *************** Send Transaction **********************
+ // *******************************************************
+
+async function sendTransaction() {
+ let v_flags = 0
+ if (clawbackSlider.checked) {v_flags+=64}
+ if (lockSlider.checked) {v_flags+=2}
+ if (authTokensSlider.checked) {v_flags +=4}
+ if (txrSlider.checked) {v_flags += 32}
+ if (tradeSlider.checked) {v_flags += 16}
+ if (escrowSlider.checked) {v_flags+=8}
+ results = 'Connecting to ' + getNet() + '....'
+ mptIssuanceIdField.value = results
+ let net = getNet()
+ const my_wallet = xrpl.Wallet.fromSeed(seedField.value)
+ const client = new xrpl.Client(net)
+ await client.connect()
+ const metadataHexString = xrpl.convertStringToHex(metadataTextArea.value)
+ const transactionJson = {
+ "TransactionType": "MPTokenIssuanceCreate",
+ "Account": accountField.value,
+ "AssetScale": parseInt(assetScaleField.value),
+ "MaximumAmount": maximumAmountField.value,
+ "TransferFee": parseInt(transferFeeField.value),
+ "Flags": v_flags,
+ "MPTokenMetadata": metadataHexString
+ }
+ const tx = await client.submitAndWait(transactionJson, { wallet: my_wallet} )
+ if (document.getElementById("tn").checked) {
+ resultField.value += "\n Success! Ledger Index: " + tx.result.ledger_index + "\nSee https://testnet.xrpl.org/ledgers/" + tx.result.ledger_index
+ } else {
+ resultField.value += "\n Success! Ledger Index: " + tx.result.ledger_index + "\nSee https://devnet.xrpl.org/ledgers/" + tx.result.ledger_index
+ }
+ mptIssuanceIdField.value = JSON.stringify(tx.result.meta.mpt_issuance_id)
+}
+// *****************************************************
+// ************ Gather MPT Info ********************
+// *****************************************************
+
+function gatherMptInfo() {
+ let mptInfo = accountNameField.value + "\n" + accountField.value + "\n" + seedField.value + "\n" + mptIssuanceIdField.value
+ resultField.value = mptInfo
+}
\ No newline at end of file
diff --git a/_code-samples/mpt-generator/mpt-generator.zip b/_code-samples/mpt-generator/mpt-generator.zip
new file mode 100644
index 00000000000..33ae6a4622f
Binary files /dev/null and b/_code-samples/mpt-generator/mpt-generator.zip differ
diff --git a/_code-samples/mpt-sender/account-support.js b/_code-samples/mpt-sender/account-support.js
new file mode 100644
index 00000000000..c898d4a2258
--- /dev/null
+++ b/_code-samples/mpt-sender/account-support.js
@@ -0,0 +1,101 @@
+// ******************************************************
+// ************* Get the Preferred Network **************
+// ******************************************************
+
+function getNet() {
+ let net
+ if (document.getElementById("tn").checked) net = "wss://s.altnet.rippletest.net:51233/"
+ if (document.getElementById("dn").checked) net = "wss://s.devnet.rippletest.net:51233/"
+ return net
+} // End of getNet()
+
+// *******************************************************
+// ************* Get Account *****************************
+// *******************************************************
+
+async function getAccount() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ let faucetHost = null
+ const my_wallet = (await client.fundWallet(null, { faucetHost})).wallet
+ const newAccount = [my_wallet.address, my_wallet.seed]
+ return (newAccount)
+ client.disconnect()
+} // End of getAccount()
+
+async function getNewAccount1() {
+ account1address.value = "Getting new account."
+ const accountInfo= await getAccount()
+ account1address.value = accountInfo[0]
+ account1seed.value = accountInfo[1]
+}
+
+async function getNewAccount2() {
+ account2address.value = "Getting new account."
+ const accountInfo= await getAccount()
+ account2address.value = accountInfo[0]
+ account2seed.value = accountInfo[1]
+}
+
+// *****************************************************
+// ********** Get Account from Seed ********************
+// *****************************************************
+
+async function getAccountFromSeed(my_seed) {
+ const net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ let results = '\nConnected, finding wallet.\n'
+ resultField.value = results
+ const wallet = xrpl.Wallet.fromSeed(my_seed)
+// ----------------------Populate the fields for left and right accounts.
+ const address = wallet.address
+ client.disconnect()
+ return (address)
+} // End of getAccountFromSeed()
+
+async function getAccountFromSeed1() {
+ account1address.value = await getAccountFromSeed(account1seed.value)
+}
+
+async function getAccountFromSeed2() {
+ account2address.value = await getAccountFromSeed(account2seed.value)
+}
+
+function gatherAccountInfo() {
+ let accountData = account1name.value + "\n" + account1address.value + "\n" + account1seed.value + "\n"
+ accountData += account2name.value + "\n" + account2address.value + "\n" + account2seed.value
+ resultField.value = accountData
+}
+
+function distributeAccountInfo() {
+ let accountInfo = resultField.value.split("\n")
+ account1name.value = accountInfo[0]
+ account1address.value = accountInfo[1]
+ account1seed.value = accountInfo[2]
+ account2name.value = accountInfo[3]
+ account2address.value = accountInfo[4]
+ account2seed.value = accountInfo[5]
+}
+
+async function getBalance() {
+ const net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ xrpBalanceField.value = await client.getXrpBalance(accountAddressField.value)
+}
+
+function populate1() {
+ accountNameField.value = account1name.value
+ accountAddressField.value = account1address.value
+ accountSeedField.value = account1seed.value
+ getBalance()
+}
+
+function populate2() {
+ accountNameField.value = account2name.value
+ accountAddressField.value = account2address.value
+ accountSeedField.value = account2seed.value
+ getBalance()
+}
diff --git a/_code-samples/mpt-sender/modular-tutorials.css b/_code-samples/mpt-sender/modular-tutorials.css
new file mode 100644
index 00000000000..1bc42daacce
--- /dev/null
+++ b/_code-samples/mpt-sender/modular-tutorials.css
@@ -0,0 +1,150 @@
+body {
+ font-family: "Inter", sans-serif;
+ padding: 20px;
+ background: #abe2ff;
+}
+
+h1 {
+ font-weight: bold;
+}
+
+td {
+ padding-left: 25px;
+ vertical-align: top;
+}
+
+input,
+button {
+ padding: 6px;
+ margin-bottom: 8px;
+ border: none
+}
+
+input:read-only {
+ background-color:rgb(11, 96, 132);
+ color:white;
+ border: 0;
+}
+
+button {
+ font-weight: bold;
+ font-family: "Work Sans", sans-serif;
+ background-color: #006aff;
+ -webkit-text-fill-color: white;
+}
+
+button:hover {
+ background-color: #0555c5;
+ cursor: pointer;
+}
+
+td {
+ vertical-align: middle;
+}
+
+/* The switch - the box around the slider */
+.switch {
+ position: relative;
+ display: inline-block;
+ width: 30px;
+ height: 16px;
+}
+
+/* Hide default HTML checkbox */
+.switch input {
+ opacity: 0;
+ width: 0;
+ height: 0;
+}
+
+/* The slider */
+.slider {
+ position: absolute;
+ cursor: pointer;
+ top: 0;
+ left: 0;
+ right: 0;
+ bottom: 0;
+ background-color: #ccc;
+ -webkit-transition: .4s;
+ transition: .4s;
+}
+
+.slider:before {
+ position: absolute;
+ content: "";
+ height: 13px;
+ width: 13px;
+ left: 4px;
+ bottom: 2px;
+ background-color: white;
+ -webkit-transition: .4s;
+ transition: .4s;
+}
+
+input:checked+.slider {
+ background-color: #2196F3;
+}
+
+input:focus+.slider {
+ box-shadow: 0 0 1px #2196F3;
+}
+
+input:checked+.slider:before {
+ -webkit-transform: translateX(13px);
+ -ms-transform: translateX(13px);
+ transform: translateX(13px);
+}
+
+/* Rounded sliders */
+.slider.round {
+ border-radius: 17px;
+}
+
+.slider.round:before {
+ border-radius: 50%;
+}
+.tooltip {
+ position: relative;
+ border-bottom: 1px dotted black;
+}
+
+.tooltip:before {
+ content: attr(tooltip-data);
+ position: absolute;
+ width: 250px;
+ background-color: #006aff;
+ color: #fff;
+ text-align: center;
+ padding: 15px;
+ line-height: 1.1;
+ border-radius: 5px;
+ z-index: 1;
+ opacity: 0;
+ transition: opacity .5s;
+ bottom: 125%;
+ left: 50%;
+ margin-left: -60px;
+ font-size: 0.70em;
+ visibility: hidden;
+}
+
+.tooltip:after {
+ content: "";
+ position: absolute;
+ bottom: 75%;
+ left: 50%;
+ margin-left: -5px;
+ border-width: 5px;
+ border-style: solid;
+ opacity: 0;
+ transition: opacity .5s;
+ border-color: #000 transparent transparent transparent;
+ visibility: hidden;
+}
+
+.tooltip:hover:before,
+.tooltip:hover:after {
+ opacity: 1;
+ visibility: visible;
+}
\ No newline at end of file
diff --git a/_code-samples/mpt-sender/send-mpt.html b/_code-samples/mpt-sender/send-mpt.html
new file mode 100644
index 00000000000..0f3cbbe93f0
--- /dev/null
+++ b/_code-samples/mpt-sender/send-mpt.html
@@ -0,0 +1,218 @@
+
+
+ Send MPT
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Send MPT
+
+
+
+
\ No newline at end of file
diff --git a/_code-samples/mpt-sender/send-mpt.js b/_code-samples/mpt-sender/send-mpt.js
new file mode 100644
index 00000000000..1f18870d28b
--- /dev/null
+++ b/_code-samples/mpt-sender/send-mpt.js
@@ -0,0 +1,101 @@
+// *******************************************************
+// ********************* Send MPT ************************
+// *******************************************************
+
+async function sendMPT() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ let results = `Connected to ${net}....`
+ resultField.value = results
+ const wallet = xrpl.Wallet.fromSeed(accountSeedField.value)
+ const mpt_issuance_id = mptIdField.value
+ const mpt_quantity = amountField.value
+ const send_mpt_tx = {
+ "TransactionType": "Payment",
+ "Account": wallet.address,
+ "Amount": {
+ "mpt_issuance_id": mpt_issuance_id,
+ "value": mpt_quantity,
+ },
+ "Destination": destinationField.value,
+ }
+ const pay_prepared = await client.autofill(send_mpt_tx)
+ const pay_signed = wallet.sign(pay_prepared)
+ results += `\n\nSending ${mpt_quantity} ${mpt_issuance_id} to ${destinationField.value} ...`
+ resultField.value = results
+ const pay_result = await client.submitAndWait(pay_signed.tx_blob)
+ if (pay_result.result.meta.TransactionResult == "tesSUCCESS") {
+ results += 'Transaction succeeded.\n\n'
+ results += JSON.stringify(pay_result.result, null, 2)
+ resultField.value = results
+ } else {
+ results += `\nTransaction failed: ${pay_result.result.meta.TransactionResult}\n\n`
+ results += JSON.stringify(pay_result.result, null, 2)
+ resultField.value = results
+ }
+ client.disconnect()
+} // end of sendMPT()
+
+// *******************************************************
+// ******************** Get MPTs *************************
+// *******************************************************
+
+async function getMPTs() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ const wallet = xrpl.Wallet.fromSeed(accountSeedField.value)
+ let results = `Connected to ${net}....`
+ resultField.value = results
+ const mpts = await client.request({
+ command: "account_objects",
+ account: wallet.address,
+ ledger_index: "validated",
+ type: "mptoken"
+ })
+ let JSONString = JSON.stringify(mpts.result, null, 2)
+ let JSONParse = JSON.parse(JSONString)
+ let numberOfMPTs = JSONParse.account_objects.length
+ let x = 0
+ while (x < numberOfMPTs){
+ results += "\n\nMPT Issuance ID: " + JSONParse.account_objects[x].MPTokenIssuanceID
+ + "\nMPT Amount: " + JSONParse.account_objects[x].MPTAmount
+ x++
+ }
+ results += "\n\n" + JSONString
+ resultField.value = results
+ client.disconnect()
+} // End of getMPTs()
+
+// **********************************************************************
+// ****** MPTAuthorize Transaction ***************************************
+// **********************************************************************
+
+async function authorizeMPT() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ let results = `Connected to ${net}....`
+ resultField.value = results
+ const wallet = xrpl.Wallet.fromSeed(accountSeedField.value)
+ const mpt_issuance_id = mptIdField.value
+ const auth_mpt_tx = {
+ "TransactionType": "MPTokenAuthorize",
+ "Account": wallet.address,
+ "MPTokenIssuanceID": mpt_issuance_id,
+ }
+ const auth_prepared = await client.autofill(auth_mpt_tx)
+ const auth_signed = wallet.sign(auth_prepared)
+ results += `\n\nSending authorization...`
+ resultField.value = results
+ const auth_result = await client.submitAndWait(auth_signed.tx_blob)
+ if (auth_result.result.meta.TransactionResult == "tesSUCCESS") {
+ results += `\nTransaction succeeded`
+ resultField.value = results
+ } else {
+ results += `\nTransaction failed: ${auth_result.result.meta.TransactionResult}`
+ resultField.value = results
+ }
+ client.disconnect()
+} // end of MPTAuthorize()
\ No newline at end of file
diff --git a/_code-samples/mpt-sender/send-mpt.zip b/_code-samples/mpt-sender/send-mpt.zip
new file mode 100644
index 00000000000..de937694b03
Binary files /dev/null and b/_code-samples/mpt-sender/send-mpt.zip differ
diff --git a/docs/concepts/tokens/fungible-tokens/multi-purpose-tokens.md b/docs/concepts/tokens/fungible-tokens/multi-purpose-tokens.md
index ba35763deec..6a906695ffd 100644
--- a/docs/concepts/tokens/fungible-tokens/multi-purpose-tokens.md
+++ b/docs/concepts/tokens/fungible-tokens/multi-purpose-tokens.md
@@ -55,6 +55,14 @@ MPTs are intended to be complementary to IOUs. While there might be use cases w
## See Also
+- **Use Case**
+
+ - [Creating an Asset-backed Multi-purpose Token](../../../use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md)
+
+- **Tutorial**
+
+ - [Sending MPTs](../../../tutorials/javascript/send-payments/sending-mpts.md)
+
- **References:**
- [MPToken](../../../references/protocol/ledger-data/ledger-entry-types/mptoken.md)
- [MPTokenIssuance](../../../references/protocol/ledger-data/ledger-entry-types/mptokenissuance.md)
diff --git a/docs/img/tut-send-mpt-0-empty-form.png b/docs/img/tut-send-mpt-0-empty-form.png
new file mode 100644
index 00000000000..8e13d0a1fb1
Binary files /dev/null and b/docs/img/tut-send-mpt-0-empty-form.png differ
diff --git a/docs/img/tut-send-mpt-1-gathered-info.png b/docs/img/tut-send-mpt-1-gathered-info.png
new file mode 100644
index 00000000000..d1dcf46b4f3
Binary files /dev/null and b/docs/img/tut-send-mpt-1-gathered-info.png differ
diff --git a/docs/img/tut-send-mpt-2-account-2.png b/docs/img/tut-send-mpt-2-account-2.png
new file mode 100644
index 00000000000..543bfbe114b
Binary files /dev/null and b/docs/img/tut-send-mpt-2-account-2.png differ
diff --git a/docs/img/tut-send-mpt-2-authorize-mpt.png b/docs/img/tut-send-mpt-2-authorize-mpt.png
new file mode 100644
index 00000000000..7f481e5715f
Binary files /dev/null and b/docs/img/tut-send-mpt-2-authorize-mpt.png differ
diff --git a/docs/img/tut-send-mpt-3-send-mpt.png b/docs/img/tut-send-mpt-3-send-mpt.png
new file mode 100644
index 00000000000..2e3d1a05ee1
Binary files /dev/null and b/docs/img/tut-send-mpt-3-send-mpt.png differ
diff --git a/docs/img/tut-send-mpt-4-get-mpts.png b/docs/img/tut-send-mpt-4-get-mpts.png
new file mode 100644
index 00000000000..fc86e95dbd4
Binary files /dev/null and b/docs/img/tut-send-mpt-4-get-mpts.png differ
diff --git a/docs/img/uc-mpt1-mpt-generator-empty-form.png b/docs/img/uc-mpt1-mpt-generator-empty-form.png
new file mode 100644
index 00000000000..ed779feb3d1
Binary files /dev/null and b/docs/img/uc-mpt1-mpt-generator-empty-form.png differ
diff --git a/docs/img/uc-mpt1-mpt-generator.png b/docs/img/uc-mpt1-mpt-generator.png
new file mode 100644
index 00000000000..6a7060f50ae
Binary files /dev/null and b/docs/img/uc-mpt1-mpt-generator.png differ
diff --git a/docs/img/uc-mpt1-t-bill-account-configuration.png b/docs/img/uc-mpt1-t-bill-account-configuration.png
new file mode 100644
index 00000000000..32e58895490
Binary files /dev/null and b/docs/img/uc-mpt1-t-bill-account-configuration.png differ
diff --git a/docs/img/uc-mpt1-t-bill-create-success.png b/docs/img/uc-mpt1-t-bill-create-success.png
new file mode 100644
index 00000000000..20a9b6d7667
Binary files /dev/null and b/docs/img/uc-mpt1-t-bill-create-success.png differ
diff --git a/docs/img/uc-mpt1-t-bill-gather-mpt-info.png b/docs/img/uc-mpt1-t-bill-gather-mpt-info.png
new file mode 100644
index 00000000000..3a19c786054
Binary files /dev/null and b/docs/img/uc-mpt1-t-bill-gather-mpt-info.png differ
diff --git a/docs/img/uc-mpt1-t-bill-in-explorer.png b/docs/img/uc-mpt1-t-bill-in-explorer.png
new file mode 100644
index 00000000000..150ee741583
Binary files /dev/null and b/docs/img/uc-mpt1-t-bill-in-explorer.png differ
diff --git a/docs/img/uc-mpt1-t-bill-mpt-generator-generate-code.png b/docs/img/uc-mpt1-t-bill-mpt-generator-generate-code.png
new file mode 100644
index 00000000000..fa80fc063b6
Binary files /dev/null and b/docs/img/uc-mpt1-t-bill-mpt-generator-generate-code.png differ
diff --git a/docs/img/uc-mpt1-t-bill-mpt-generator.png b/docs/img/uc-mpt1-t-bill-mpt-generator.png
new file mode 100644
index 00000000000..9bd127265c1
Binary files /dev/null and b/docs/img/uc-mpt1-t-bill-mpt-generator.png differ
diff --git a/docs/tutorials/javascript/send-payments/sending-mpts.md b/docs/tutorials/javascript/send-payments/sending-mpts.md
new file mode 100644
index 00000000000..52cfa2c32bb
--- /dev/null
+++ b/docs/tutorials/javascript/send-payments/sending-mpts.md
@@ -0,0 +1,824 @@
+---
+seo:
+ description: Issue an asset-backed token such as a US Treasury bill using multi-purpose tokens.
+labels:
+ - Tokens
+ - MPT
+---
+# Sending MPTs
+
+_(Requires the [MPToken amendment][] {% not-enabled /%})_
+
+To send an MPT to another account, the receiving account must first authorize the receipt of the MPT, based on its MPToken Issuance ID. This is to prevent malicious users from spamming accounts with unwanted tokens that could negatively impact storage and XRP reserves.
+
+Once an account receives an MPT, it can send the MPT to another account, provided the MPT was created with the _Can Transfer_ flag set, and the receiving account authorizes the MPT.
+
+## Send MPT Utility
+
+The Send MPT utility lets you create an account, authorize it to receive a specific MPT issuance, then send it the authorized MPT from an issuer or holder account. (You can issue an MPT using the [MPT Generator](../../../use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md) utility.)
+
+
+
+You can download a [standalone version of the MPT Sender](../../../../_code-samples/mpt-sender/send-mpt.zip) as sample code.
+
+## Get Accounts
+
+To send an MPT, you need the **Seed** value for the MPT issuer to retrieve its account, then you need either a new account or an account seed for the target account. You can use the [MPT Generator](../../../use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md) to create a new MPT for transfer.
+
+To get the accounts:
+
+1. Open send-mpt.html in a browser.
+2. Choose your ledger instance (**Devnet** or **Testnet**).
+3. If you used the MPT Generator:
+ 1. Paste the gathered info in the **Result** field.
+ 
+ 2. Cut and paste the MPT Issuance ID to the **MPT Issuance ID** field.
+ 3. Click **Distribute Account Info** to populate the **Account 1** fields.
+ (If you did not use the MPT Generator, enter the **Account 1 Name**, **Account 1 Address**, **Account 1 Seed**, and **MPT Issuance ID** in the corresponding fields.)
+4. Click **Get New Account 2**.
+5. Optionally, add the **Account 2 Name**, an arbitrary human-readable name that helps to differentiate the accounts.
+
+
+## Authorize MPT
+
+To receive MPTs, an account needs to authorize the MPT.
+
+To authorize Account 2 to accept MPTs:
+
+1. Click the **Account 2** radio button.
+2. Enter an **Amount**, the maximum number of MPTs the account will accept.
+2. Click **Authorize MPTs**.
+
+
+## Send MPT
+
+To send an MPT:
+
+1. Click the **Account 1** radio button.
+3. Enter the **MPT Issuance ID**.
+2. Enter an **Amount** of MPTs to send.
+3. Enter the **Destination** (likely the value in the **Account 2 Address** field, but it can be any account on the same ledger instance).
+4. Click **Send MPT**.
+
+
+## Get MPTs
+
+To verify receipt of the MPTs:
+
+1. Click the **Account 2** radio button.
+2. Click **Get MPTs**.
+
+
+# Code Walkthrough
+
+You can download a [standalone version of the MPT Sender](../../../../_code-samples/mpt-sender/send-mpt.zip) as sample code.
+
+## send-mpt.js
+
+The code that supports the MPT features is in the `send-mpt.js` file. Standard support for connecting to the XRP Ledger is included in the `account-support.js` file.
+
+### sendMPT()
+
+Connect to the network and instantiate the account wallet.
+
+```javascript
+async function sendMPT() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ let results = `Connected to ${net}....`
+ resultField.value = results
+ const wallet = xrpl.Wallet.fromSeed(accountSeedField.value)
+```
+
+Instantiate the parameter variables.
+
+```javascript
+ const mpt_issuance_id = mptIdField.value
+ const mpt_quantity = amountField.value
+```
+
+Create a Payment transaction using the MPT for the Amount.
+
+```javascript
+ const send_mpt_tx = {
+ "TransactionType": "Payment",
+ "Account": wallet.address,
+ "Amount": {
+ "mpt_issuance_id": mpt_issuance_id,
+ "value": mpt_quantity,
+ },
+ "Destination": destinationField.value,
+ }
+```
+
+Prepare and sign the transaction.
+
+```javascript
+ const pay_prepared = await client.autofill(send_mpt_tx)
+ const pay_signed = wallet.sign(pay_prepared)
+```
+
+Send the prepared transaction and report the results.
+
+```javascript
+ results += `\n\nSending ${mpt_quantity} ${mpt_issuance_id} to ${destinationField.value} ...`
+ resultField.value = results
+ const pay_result = await client.submitAndWait(pay_signed.tx_blob)
+ if (pay_result.result.meta.TransactionResult == "tesSUCCESS") {
+ results += 'Transaction succeeded.\n\n'
+ results += JSON.stringify(pay_result.result, null, 2)
+ resultField.value = results
+ } else {
+ results += `\nTransaction failed: ${pay_result.result.meta.TransactionResult}\n\n`
+ results += JSON.stringify(pay_result.result, null, 2)
+ resultField.value = results
+ }
+ client.disconnect()
+} // end of sendMPT()
+```
+
+## getMPTs
+
+Get all of the MPTs for the selected account by filtering for MPT objects and looping through the array to display them one at a time.
+
+Connect to the XRPL instance and get the account wallet.
+
+```javascript
+async function getMPTs() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ const wallet = xrpl.Wallet.fromSeed(accountSeedField.value)
+ let results = `Connected to ${net}....`
+ resultField.value = results
+```
+
+Send an `account_objects` request, specifying the type _mptoken_.
+
+```javascript
+ const mpts = await client.request({
+ command: "account_objects",
+ account: wallet.address,
+ ledger_index: "validated",
+ type: "mptoken"
+ })
+```
+
+Stringify and parse the JSON result string.
+
+```javascript
+ let JSONString = JSON.stringify(mpts.result, null, 2)
+ let JSONParse = JSON.parse(JSONString)
+ let numberOfMPTs = JSONParse.account_objects.length
+```
+
+Loop through the filtered array of account_objects to list all of the MPTs held by the account.
+
+```javascript
+ let x = 0
+ while (x < numberOfMPTs){
+ results += "\n\nMPT Issuance ID: " + JSONParse.account_objects[x].MPTokenIssuanceID
+ + "\nMPT Amount: " + JSONParse.account_objects[x].MPTAmount
+ x++
+ }
+```
+
+Return the parsed results, followed by the raw results.
+
+```javascript
+ results += "\n\n" + JSONString
+ resultField.value = results
+ client.disconnect()
+} // End of getMPTs()
+```
+
+## authorizeMPT
+
+Before you can send an MPT to another account, the target account must authorize the MPT.
+
+Connect to the XRPL and instantiate the account wallet.
+
+```javascript
+async function authorizeMPT() {
+ let net = getNet()
+ const client = new xrpl.Client(net)
+ await client.connect()
+ let results = `Connected to ${net}....`
+ resultField.value = results
+ const wallet = xrpl.Wallet.fromSeed(accountSeedField.value)
+```
+
+Capture the MPT issuance ID in a variable.
+
+```javascript
+ const mpt_issuance_id = mptIdField.value
+```
+
+Create the MPTokenAuthorize transaction, passing the target account's address and the MPT Issuance ID.
+
+```javascript
+ const auth_mpt_tx = {
+ "TransactionType": "MPTokenAuthorize",
+ "Account": wallet.address,
+ "MPTokenIssuanceID": mpt_issuance_id,
+ }
+```
+
+Prepare, sign, and send the transaction.
+
+```javascript
+ const auth_prepared = await client.autofill(auth_mpt_tx)
+ const auth_signed = wallet.sign(auth_prepared)
+ results += `\n\nSending authorization...`
+ resultField.value = results
+ const auth_result = await client.submitAndWait(auth_signed.tx_blob)
+```
+
+Report the results.
+
+```javascript
+ if (auth_result.result.meta.TransactionResult == "tesSUCCESS") {
+ results += `\nTransaction succeeded`
+ resultField.value = results
+ } else {
+ results += `\nTransaction failed: ${auth_result.result.meta.TransactionResult}`
+ resultField.value = results
+ }
+ client.disconnect()
+} // end of MPTAuthorize()
+
+```
+
+## send-mpt.html
+
+```html
+
+
+ Send MPT
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Send MPT
+
+
+
+
+```
+
+{% raw-partial file="/docs/_snippets/common-links.md" /%}
\ No newline at end of file
diff --git a/docs/use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md b/docs/use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md
new file mode 100644
index 00000000000..b5c0972b54f
--- /dev/null
+++ b/docs/use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md
@@ -0,0 +1,455 @@
+---
+seo:
+ description: Issue an asset-backed token such as a US Treasury bill using multi-purpose tokens.
+labels:
+ - Tokens
+ - MPT
+---
+# Creating an Asset-backed Multi-purpose Token
+
+_(Requires the [MPToken amendment][] {% not-enabled /%})_
+
+_As a financial professional, I want to use multi-purpose tokens to create an asset-backed token in order to profit from resale transactions._
+
+A multi-purpose token (MPT) is a compact and flexible object that offers the best aspects of fungible and non-fungible tokens. It is the next generation of tokenization on the XRPL. Notable features include:
+
+- MPTs store metadata directly on the XRPL blockchain, with the option of linking to additional off-chain data.
+- MPTs can have a fixed token supply, with a cap on the maximum number of tokens.
+- MPT issuers can collect transfer fees each time the token is traded.
+- MPTs can be non-transferable, for use cases such as airline credits.
+- MPTs also allow advanced compliance features.
+
+To learn more, see [Multi-purpose Tokens](../../concepts/tokens/fungible-tokens/multi-purpose-tokens.md).
+
+## MPT Generator
+
+
+
+You can download a [standalone version of the MPT Generator](../../../_code-samples/mpt-generator/mpt-generator.zip) as sample code.
+
+In practice, you want to use an Issuer account configuration to issue an MPT, but you can try the form with a new, unconfigured account and the transaction works fine. A T-bill is one example of the many types of asset you can create and trade on the XRP Ledger.
+
+
+## Creating a US Treasury Bill as an MPT
+
+A US Treasury bill (T-bill) is a short-term debt security issued by the US government. T-bills are considered a safe investment because they're backed by the US government. T-bills are appealing to investors in American states that have high income tax because the interest earned is exempt from state and local taxes. See [Treasury Bills In Depth](https://www.treasurydirect.gov/research-center/history-of-marketable-securities/bills/t-bills-indepth/).
+
+### Creating an Issuing Account
+
+You can use the Account Configurator to experiment with the settings for a T-bill issuing account in a sandbox environment. When you are satisfied with your configuration, you can create an account on XRPL Mainnet to begin trading.
+
+To create a new MPT Issuer account:
+- In the Account Configurator utility, choose ledger instance **Devnet**.
+- Click **Get New Account**.
+- Choose account configuration template **Issuer**.
+
+The form sets the standard flags for an Issuer account and displays additional configuration fields.
+
+
+
+| Parameter | Value | Description |
+|-----------|-------|-------------|
+| **Domain** | _TOML domain_ | URL to the server where your TOML file is stored. For an experimental account, you can enter any URL. When you configure the account, the domain is automatically converted to a hexidecimal string. If you reconfigure the account, you need to enter the original domain again. |
+| **Transfer Rate** | 1005000000 | A value between 1000000000 and 2000000000 representing the percentage value you collect when a holder transfers one of your issued tokens. A Transfer Rate of 1005000000 returns .5% of the transfer value to this account. See [Transfer Rate](../../references/protocol/transactions/types/accountset.md#transferrate). |
+| **Tick Size** | 5 | Round offers to this many significant digits. Valid values are 3 to 15, or 0 to disable. |
+| **Signer Accounts** | _account addresses_ | Accounts that have a vote regarding approval of transactions for this account. |
+| **Signer Weights** | _int_ | The weight of each signer's signature, relative to other signers. |
+| **Signer Quorum** | _int_ | The required minimum value of signer weights to approve a transaction. |
+
+{% admonition type="info" name="Note" %}In practice, configuring signers for your issuing account is a best practice. To reduce complexity, this example does not use signer configuration.{% /admonition %}
+
+#### Issuer Account Flag Settings
+
+Use the sliders to configure the standard suggested flag settings. Overall, you want holders of your T-bill to be able to trade with other holders, so rippling is an essential function. You want to be careful about what other accounts are able to send to your account, so you should disallow most types of transfers. One exception is trust lines, which you do want other accounts to be able to create to your issuing account.
+
+|Flag |Purpose |
+|------------------------------|------------------------------------------------------------------------------------|
+| defaultRipple | Allow transfers to third-party holders by default. |
+| depositAuth | Require authorization for another account to deposit to this account. |
+| disallowIncomingCheck | Prevent other accounts from sending checks to this account. |
+| disallowIncomingNFTokenOffer | Prevent other accounts from sending NFTokenOffers to this account. |
+| disallowIncomingPayChan | Prevent other accounts from creating Payment Channels to this account. |
+| disallowIncomingXRP | Prevent other accounts from sending XRP to this account. |
+
+Once you've set your preferred values in the account configurator, click **Configure Account** to finish preparing your account to issue T-bills.
+
+### Generating a US Treasury Bill as a Multi-purpose Token
+
+You can represent a US Treasury Bill (T-bill) on the XRPL by creating a multi-purpose token. The example below shows random possible values you might use as a starting point. Adjust as needed for your use case.
+
+
+
+| Parameter | Value | Description |
+|-----------|-------|-------------|
+| Asset Scale | 2 | The difference, in orders of magnitude, between a standard T-bill unit and the corresponding fractional unit. |
+| Maximum Tokens | 500000 | The maximum number of this T-bill MPT that will ever be issued by this account. |
+| Transfer Fee | 314 | Fee collected when the T-bill MPT is transferred to another account. |
+
+As an example, you might set the following flags when creating your T-bill MPT.
+
+| Flag | Description |
+|------|-------------|
+| Can Transfer | A holder can transfer the T-bill MPT to another account. |
+| Can Trade | A holder can trade the T-bill MPT with another account. |
+
+#### Token Metadata
+
+The metadata you provide is what distinguishes your token from other MPTs. The following JSON definition shows sample configuration information for a T-bill. Copy and paste the data, or your own metadata, into the **Token Metadata** field.
+
+```json
+{
+ "Name": "US Treasury Bill Token",
+ "Identifier": "USTBT",
+ "Issuer": "US Treasury",
+ "IssueDate": "2024-03-25",
+ "MaturityDate": "2025-03-25",
+ "FaceValue": 1000,
+ "InterestRate": 2.5,
+ "InterestFrequency": "Quarterly",
+ "Collateral": "US Government",
+ "Jurisdiction": "United States",
+ "RegulatoryCompliance": "SEC Regulations",
+ "SecurityType": "Treasury Bill",
+ "ExternalUrl": "https://example.com/t-bill-token-metadata.json"
+ }
+```
+Once you've set your preferred values, click **Generate Transaction** to see the transaction syntax for your settings. The `Flags` field displays the sum of the flags you've selected, and the `MPTokenMetadata` is converted to a hexidecimal string.
+
+
+
+To create your T-bill MPT, click **Send Transaction**. When your transaction succeeds, a link to the record in the XRPL Explorer is displayed in the result field.
+
+
+
+Follow the link and scroll down to find the `MPTokenIssuanceCreate` transaction for your new T-bill in the Explorer.
+
+
+
+Click **Gather MPT Information** to copy the account information and MPT Issuance ID to the result field. Copy the information and save it to send the MPT to another account as shown in [Sending MPTs](../../tutorials/javascript/send-payments/sending-mpts.md).
+
+
+
+{% raw-partial file="/docs/_snippets/common-links.md" /%}
\ No newline at end of file
diff --git a/docs/use-cases/tokenization/index.page.tsx b/docs/use-cases/tokenization/index.page.tsx
index 40734440f3b..16e8cba9728 100644
--- a/docs/use-cases/tokenization/index.page.tsx
+++ b/docs/use-cases/tokenization/index.page.tsx
@@ -20,6 +20,10 @@ const useCases = [
description: "NFT Marketplace",
link: "/docs/use-cases/tokenization/nftoken-marketplace/",
},
+ {
+ description: "Multi-purpose Token Issuer",
+ link: "/docs/use-cases/tokenization/creating-an-asset-backed-multi-purpose-token"
+ },
{
description: "Authorized Minter",
link: "/docs/use-cases/tokenization/authorized-minter/",
diff --git a/docs/use-cases/tokenization/nft-mkt-overview.md b/docs/use-cases/tokenization/nft-mkt-overview.md
index bc00db1b499..90b7cec1174 100644
--- a/docs/use-cases/tokenization/nft-mkt-overview.md
+++ b/docs/use-cases/tokenization/nft-mkt-overview.md
@@ -1,6 +1,4 @@
---
-html: nft-mkt-overview.html
-parent: tokenization.html
seo:
description: Overview of NFT Marketplace use cases.
labels:
diff --git a/sidebars.yaml b/sidebars.yaml
index 4c6084c22ca..1e2bb639691 100644
--- a/sidebars.yaml
+++ b/sidebars.yaml
@@ -33,6 +33,7 @@
- page: docs/use-cases/tokenization/authorized-minter.md
- page: docs/use-cases/tokenization/digital-artist.md
- page: docs/use-cases/tokenization/real-world-assets.page.tsx
+ - page: docs/use-cases/tokenization/creating-an-asset-backed-multi-purpose-token.md
- page: docs/use-cases/defi/index.md
expanded: false
items:
@@ -188,6 +189,7 @@
- page: docs/tutorials/javascript/send-payments/create-time-based-escrows.md
- page: docs/tutorials/javascript/send-payments/create-conditional-escrows.md
- page: docs/tutorials/javascript/send-payments/send-and-cash-checks.md
+ - page: docs/tutorials/javascript/send-payments/sending-mpts.md
- page: docs/tutorials/javascript/nfts/index.md
expanded: false
items:
| |