token script study
https://www.tokenscript.org/
https://community.tokenscript.org/
https://medium.com/alphawallet/tokenscript/home
https://github.com/TokenScript/TokenScript/tree https://github.com/TokenScript/TokenScriptTestContracts/tree/master https://github.com/TokenScript/TokenScript-Examples/tree/master/tutorial
EntryToken.sol
pragma solidity ^0.4.25; //
contract EntryToken
{
mapping(address => uint256[]) inventory;
uint256[] public spawnedTokens;
mapping(bytes32 => bool) signatureChecked;
address public organiser;
address public paymaster;
string public name;
uint8 public constant decimals = 0; //no decimals as Tokens cannot be split
bool expired = false;
string public state;
string public locality;
string public street;
string public building;
string public symbol;
bytes4 balHash = bytes4(keccak256('balanceOf(address)'));
bytes4 tradeHash =
bytes4(keccak256('trade(uint256,uint256[],uint8,bytes32,bytes32)'));
bytes4 passToHash =
bytes4(keccak256('passTo(uint256,uint256[],uint8,bytes32,bytes32,address)'));
bytes4 spawnPassToHash =bytes4(keccak256('spawnPassTo(uint256,uint256[],uint8,bytes32,bytes32,address)'
));
event Transfer(address indexed _to, uint256 count);
event TransferFrom(address indexed _from, address indexed _to, uint256
count);
event Trade(address indexed seller, uint256[] TokenIndices, uint8 v, bytes32
r, bytes32 s);
event PassTo(uint256[] TokenIndices, uint8 v, bytes32 r, bytes32 s, address
indexed recipient);
modifier organiserOnly()
{
require(msg.sender == organiser);
_;
}
modifier payMasterOnly()
{
require(msg.sender == paymaster);
_;
}
function() payable public { revert(); } //should not send any ether directly
constructor (
uint256[] Tokens,
string buildingName,
string streetName,
string localityName,
string stateName,
string symbolName,
string contractName) public
{
organiser = msg.sender;
paymaster = msg.sender;
inventory[msg.sender] = Tokens;
building = buildingName;
street = streetName;
locality = localityName;
state = stateName;
symbol = symbolName;
name = contractName;
}
function supportsInterface(bytes4 interfaceID) external view returns (bool)
{
if(interfaceID == balHash
|| interfaceID == tradeHash
|| interfaceID == passToHash
|| interfaceID == spawnPassToHash) return true;
return false;
}
function isExpired(uint256 tokenId) public view returns(bool)
{
return expired;
}
function getStreet(uint256 tokenId) public view returns(string)
{
return street;
}
function getBuildingName(uint256 tokenId) public view returns(string)
{
return building;
}
function getState(uint256 tokenId) public view returns(string)
{
return state;
}
function getLocality(uint256 tokenId) public view returns(string)
{
return locality;
}
function getDecimals() public pure returns(uint)
{
return decimals;
}
function name() public view returns(string)
{
return name;
}
function setExpired(uint256[] tokenIds, bool isExpired) public organiserOnly
{
expired = isExpired;
}
function setStreet(uint256[] tokenIds, string newStreet) public
organiserOnly returns(string)
{
street = newStreet;
}
function setBuilding(uint256[] tokenIds, string newBuildingName) public
organiserOnly returns(string)
{
building = newBuildingName;
}
function setState(uint256[] tokenIds, string newState) public organiserOnly
returns(string)
{
state = newState;
}
function setLocality(uint256[] tokenIds, string newLocality) public organiserOnly returns(string)
{
locality = newLocality;
}
// example: 0, [3, 4], 27, "0x9CAF1C785074F5948310CD1AA44CE2EFDA0AB19C308307610D7BA2C74604AE98", "0x23D8D97AB44A2389043ECB3C1FB29C40EC702282DB6EE1D2B2204F8954E4B451"
// price is encoded in the server and the msg.value is added to the message digest,
// if the message digest is thus invalid then either the price or something else in the message is invalid
function trade(uint256 expiry,
uint256[] TokenIndices,
uint8 v,
bytes32 r,
bytes32 s) public payable
{
//checks expiry timestamp,
//if fake timestamp is added then message verification will fail
require(expiry > block.timestamp || expiry == 0);
bytes32 message = encodeMessage(msg.value, expiry, TokenIndices);
address seller = ecrecover(message, v, r, s);
require(seller == organiser); //only contract owner can issue magiclinks
for(uint i = 0; i < TokenIndices.length; i++)
{ // transfer each individual Tokens in the ask order
uint256 index = TokenIndices[i];
assert(inventory[seller][index] != uint256(0)); // 0 means Token gone.
inventory[msg.sender].push(inventory[seller][index]);
// 0 means Token gone.
delete inventory[seller][index];
}
seller.transfer(msg.value);
emit Trade(seller, TokenIndices, v, r, s);
}
function loadNewTokens(uint256[] Tokens) public organiserOnly
{
for(uint i = 0; i < Tokens.length; i++)
{
inventory[organiser].push(Tokens[i]);
}
}
//for new Tokens to be created and given over
//this requires a special magic link format with tokenids inside rather than indicies
function spawnPassTo(uint256 expiry,
uint256[] Tokens,
uint8 v,
bytes32 r,
bytes32 s,
address recipient) public payable
{
require(expiry > block.timestamp || expiry == 0);
bytes32 message = encodeMessageSpawnable(msg.value, expiry, Tokens);
address giver = ecrecover(message, v, r, s);
//only the organiser can authorise this
require(giver == organiser);
require(!signatureChecked[s]);
organiser.transfer(msg.value);
for(uint i = 0; i < Tokens.length; i++)
{
inventory[recipient].push(Tokens[i]);
//log each spawned Token used for a record
spawnedTokens.push(Tokens[i]);
}
//prevent link being reused with the same signature
signatureChecked[s] = true;
}
//check if a spawnable Token that created in a magic link is redeemed
function spawned(uint256 Token) public view returns (bool)
{
for(uint i = 0; i < spawnedTokens.length; i++)
{
if(spawnedTokens[i] == Token)
{
return true;
}
}
return false;
}
function passTo(uint256 expiry,
uint256[] TokenIndices,
uint8 v,
bytes32 r,
bytes32 s,
address recipient) public organiserOnly
{
require(expiry > block.timestamp || expiry == 0);
bytes32 message = encodeMessage(0, expiry, TokenIndices);
address giver = ecrecover(message, v, r, s);
for(uint i = 0; i < TokenIndices.length; i++)
{
uint256 index = TokenIndices[i];
//needs to use revert as all changes should be reversed
//if the user doesnt't hold all the Tokens
assert(inventory[giver][index] != uint256(0));
uint256 Token = inventory[giver][index];
inventory[recipient].push(Token);
delete inventory[giver][index];
}
emit PassTo(TokenIndices, v, r, s, recipient);
}
// Pack value, expiry, Tokens into 1 array
function encodeMessage(uint value, uint expiry, uint256[] TokenIndices)
internal view returns (bytes32)
{
bytes memory message = new bytes(84 + TokenIndices.length * 2);
address contractAddress = getThisContractAddress();
for (uint i = 0; i < 32; i++)
{
message[i] = byte(bytes32(value << (8 * i)));
}
for (i = 0; i < 32; i++)
{
message[i + 32] = byte(bytes32(expiry << (8 * i)));
}
for(i = 0; i < 20; i++)
{
message[64 + i] = byte(bytes20(contractAddress) << (8 * i));
}
for (i = 0; i < TokenIndices.length; i++)
{
message[84 + i * 2 ] = byte(TokenIndices[i] >> 8);
message[84 + i * 2 + 1] = byte(TokenIndices[i]);
}
return keccak256(message);
}
// Pack value, expiry, Tokens into 1 array
function encodeMessageSpawnable(uint value, uint expiry, uint256[] Tokens)
internal view returns (bytes32)
{
bytes memory message = new bytes(84 + Tokens.length * 32);
address contractAddress = getThisContractAddress();
for (uint i = 0; i < 32; i++)
{
message[i] = byte(bytes32(value << (8 * i)));
}
for (i = 0; i < 32; i++)
{
message[i + 32] = byte(bytes32(expiry << (8 * i)));
}
for(i = 0; i < 20; i++)
{
message[64 + i] = byte(bytes20(contractAddress) << (8 * i));
}
for (i = 0; i < Tokens.length; i++)
{
for (uint j = 0; j < 32; j++)
{
message[84 + i * 32 + j] = byte(bytes32(Tokens[i] << (8 * j)));
}
}
return keccak256(message);
}
function getSymbol() public view returns(string)
{
return symbol;
}
function balanceOf(address _owner) public view returns (uint256[])
{
return inventory[_owner];
}
function myBalance() public view returns(uint256[])
{
return inventory[msg.sender];
}
function transfer(address _to, uint256[] TokenIndices) organiserOnly public
{
for(uint i = 0; i < TokenIndices.length; i++)
{
uint index = uint(TokenIndices[i]);
require(inventory[msg.sender][index] != uint256(0));
//pushes each element with ordering
inventory[_to].push(inventory[msg.sender][index]);
delete inventory[msg.sender][index];
}
emit Transfer(_to, TokenIndices.length);
}
function transferFrom(address _from, address _to, uint256[] TokenIndices)
organiserOnly public
{
for(uint i = 0; i < TokenIndices.length; i++)
{
uint index = uint(TokenIndices[i]);
require(inventory[_from][index] != uint256(0));
//pushes each element with ordering
inventory[_to].push(inventory[_from][index]);
delete inventory[_from][index];
}
emit TransferFrom(_from, _to, TokenIndices.length);
}
function endContract() public organiserOnly
{
selfdestruct(organiser);
}
function isStormBirdContract() public pure returns (bool)
{
return true;
}
function getThisContractAddress() public view returns(address)
{
return this;
}
}
EntryToken.xml:
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<ts:token xmlns:ethereum="urn:ethereum:constantinople"
xmlns:ts="http://tokenscript.org/2020/06/tokenscript"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
name="entrytoken"
xsi:schemaLocation="http://tokenscript.org/2020/06/tokenscript http://tokenscript.org/2020/06/tokenscript.xsd"
>
<ts:label>
<ts:plurals xml:lang="en">
<ts:string quantity="one">Ticket</ts:string>
<ts:string quantity="other">Tickets</ts:string>
</ts:plurals>
<ts:plurals xml:lang="es">
<ts:string quantity="one">Boleto de admisión</ts:string>
<ts:string quantity="other">Boleto de admisiónes</ts:string>
</ts:plurals>
<ts:plurals xml:lang="zh">
<ts:string quantity="one">入場券</ts:string>
<ts:string quantity="other">入場券</ts:string>
</ts:plurals>
</ts:label>
<ts:contract interface="erc875" name="EntryToken">
<ts:address network="1">0x63cCEF733a093E5Bd773b41C96D3eCE361464942</ts:address>
<ts:address network="3">0xFB82A5a2922A249f32222316b9D1F5cbD3838678</ts:address>
<ts:address network="4">0x59a7a9fd49fabd07c0f8566ae4be96fcf20be5e1</ts:address>
<ts:address network="42">0x2B58A9403396463404c2e397DBF37c5EcCAb43e5</ts:address>
</ts:contract>
<ts:origins>
<!-- Define the contract which holds the token that the user will use -->
<ts:ethereum contract="EntryToken"></ts:ethereum>
</ts:origins>
<ts:selection filter="expired=TRUE" name="expired">
<ts:label>
<ts:plurals xml:lang="en">
<ts:string quantity="one">Expired Ticket</ts:string>
<ts:string quantity="other">Expired Tickets</ts:string>
</ts:plurals>
<ts:string xml:lang="zh">已经过期的票</ts:string>
</ts:label>
</ts:selection>
<ts:cards>
<ts:card name="main" type="token">
<ts:item-view xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">
<style type="text/css">.ts-count {
font-family: "SourceSansPro";
font-weight: bolder;
font-size: 21px;
color: rgb(117, 185, 67);
}
.ts-category {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 21px;
color: rgb(67, 67, 67);
}
.ts-venue {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 16px;
color: rgb(67, 67, 67);
}
.ts-date {
font-family: "SourceSansPro";
font-weight: bold;
font-size: 14px;
color: rgb(112, 112, 112);
margin-left: 7px;
margin-right: 7px;
}
.ts-time {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 16px;
color: rgb(112, 112, 112);
}
html {
}
body {
padding: 0px;
margin: 0px;
}
div {
margin: 0px;
padding: 0px;
}
.data-icon {
height:16px;
vertical-align: middle
}
.tbml-count { font-family: "SourceSansPro"; font-weight: bolder; font-size: 21px; color: rgb(117, 185, 67);}.tbml-category { font-family: "SourceSansPro"; font-weight: lighter; font-size: 21px; color: rgb(67, 67, 67);}.tbml-venue { font-family: "SourceSansPro"; font-weight: lighter; font-size: 16px; color: rgb(67, 67, 67);}.tbml-date { font-family: "SourceSansPro"; font-weight: bold; font-size: 14px; color: rgb(112, 112, 112); margin-left: 7px; margin-right: 7px;}.tbml-time { font-family: "SourceSansPro"; font-weight: lighter; font-size: 16px; color: rgb(112, 112, 112);} html { } body { padding: 0px; margin: 0px; } div { margin: 0px; padding: 0px; } .data-icon { height:16px; vertical-align: middle }
</style>
<body><div>
<!-- Iconified view displayed on the first page when clicking on token card in AlphaWallet -->
<p>Enter Satoshi's villa with this special token!
<img alt="" src=""></img>
</p>
</div>
</body>
</ts:item-view>
<ts:view xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">
<style type="text/css">.ts-count {
font-family: "SourceSansPro";
font-weight: bolder;
font-size: 21px;
color: rgb(117, 185, 67);
}
.ts-category {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 21px;
color: rgb(67, 67, 67);
}
.ts-venue {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 16px;
color: rgb(67, 67, 67);
}
.ts-date {
font-family: "SourceSansPro";
font-weight: bold;
font-size: 14px;
color: rgb(112, 112, 112);
margin-left: 7px;
margin-right: 7px;
}
.ts-time {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 16px;
color: rgb(112, 112, 112);
}
html {
}
body {
padding: 0px;
margin: 0px;
}
div {
margin: 0px;
padding: 0px;
}
.data-icon {
height:16px;
vertical-align: middle
}
.tbml-count { font-family: "SourceSansPro"; font-weight: bolder; font-size: 21px; color: rgb(117, 185, 67);}.tbml-category { font-family: "SourceSansPro"; font-weight: lighter; font-size: 21px; color: rgb(67, 67, 67);}.tbml-venue { font-family: "SourceSansPro"; font-weight: lighter; font-size: 16px; color: rgb(67, 67, 67);}.tbml-date { font-family: "SourceSansPro"; font-weight: bold; font-size: 14px; color: rgb(112, 112, 112); margin-left: 7px; margin-right: 7px;}.tbml-time { font-family: "SourceSansPro"; font-weight: lighter; font-size: 16px; color: rgb(112, 112, 112);} html { } body { padding: 0px; margin: 0px; } div { margin: 0px; padding: 0px; } .data-icon { height:16px; vertical-align: middle }
</style>
<script type="text/javascript">//
(function() {
'use strict'
function GeneralizedTime(generalizedTime) {
this.rawData = generalizedTime;
}
GeneralizedTime.prototype.getYear = function () {
return parseInt(this.rawData.substring(0, 4), 10);
};
GeneralizedTime.prototype.getMonth = function () {
return parseInt(this.rawData.substring(4, 6), 10) - 1;
};
GeneralizedTime.prototype.getDay = function () {
return parseInt(this.rawData.substring(6, 8), 10)
};
GeneralizedTime.prototype.getHours = function () {
return parseInt(this.rawData.substring(8, 10), 10)
};
GeneralizedTime.prototype.getMinutes = function () {
var minutes = parseInt(this.rawData.substring(10, 12), 10)
if (minutes) return minutes
return 0
};
GeneralizedTime.prototype.getSeconds = function () {
var seconds = parseInt(this.rawData.substring(12, 14), 10)
if (seconds) return seconds
return 0
};
GeneralizedTime.prototype.getMilliseconds = function () {
var startIdx
if (time.indexOf('.') !== -1) {
startIdx = this.rawData.indexOf('.') + 1
} else if (time.indexOf(',') !== -1) {
startIdx = this.rawData.indexOf(',') + 1
} else {
return 0
}
var stopIdx = time.length - 1
var fraction = '0' + '.' + time.substring(startIdx, stopIdx)
var ms = parseFloat(fraction) * 1000
return ms
};
GeneralizedTime.prototype.getTimeZone = function () {
let time = this.rawData;
var length = time.length
var symbolIdx
if (time.charAt(length - 1 ) === 'Z'){
return 0
}
if (time.indexOf('+') !== -1) {
symbolIdx = time.indexOf('+')
} else if (time.indexOf('-') !== -1) {
symbolIdx = time.indexOf('-')
} else {
return NaN
}
var minutes = time.substring(symbolIdx + 2)
var hours = time.substring(symbolIdx + 1, symbolIdx + 2)
var one = (time.charAt(symbolIdx) === '+') ? 1 : -1
var intHr = one * parseInt(hours, 10) * 60 * 60 * 1000
var intMin = one * parseInt(minutes, 10) * 60 * 1000
var ms = minutes ? intHr + intMin : intHr
return ms
};
if (typeof exports === 'object') {
module.exports = GeneralizedTime
} else if (typeof define === 'function') {
define(GeneralizedTime)
} else {
window.GeneralizedTime = GeneralizedTime
}
}())
class Token {
constructor(tokenInstance) {
this.props = tokenInstance;
}
formatGeneralizedTimeToDate(str) {
const d = new GeneralizedTime(str);
return new Date(d.getYear(), d.getMonth(), d.getDay(), d.getHours(), d.getMinutes(), d.getSeconds()).toLocaleDateString();
}
formatGeneralizedTimeToTime(str) {
const d = new GeneralizedTime(str);
return new Date(d.getYear(), d.getMonth(), d.getDay(), d.getHours(), d.getMinutes(), d.getSeconds()).toLocaleTimeString([], {hour: '2-digit', minute:'2-digit'});
}
render() {
let time;
let date;
if (this.props.time == null) {
time = "";
date = "";
} else {
time = this.formatGeneralizedTimeToTime(this.props.time.generalizedTime);
date = this.props.time == null ? "": this.formatGeneralizedTimeToDate(this.props.time.generalizedTime);
}
return `<div>
<div>
<span class="ts-count">x${this.props._count}</span> <span class="ts-category">${this.props.label}</span>
</div>
<div>
<span class="ts-venue">${this.props.building}</span>
</div>
<div style="margin: 0px; padding:0px; clear: both; height: 6px">
&nbsp;
</div>
<div>
<img src="" class="data-icon"/>
<span class="ts-date">${date}</span>
</div>
<div>
<span class="ts-time">${time}, ${this.props.locality}</span>
</div>
</div>`;
}
}
web3.tokens.dataChanged = (oldTokens, updatedTokens, tokenCardId) => {
const currentTokenInstance = updatedTokens.currentInstance;
document.getElementById(tokenCardId).innerHTML = new Token(currentTokenInstance).render();
};
//
</script>
</ts:view>
</ts:card>
<ts:card exclude="expired" name="enter" type="action">
<!-- this action is of the model confirm-back.
It should be <ts:card type="action" model="confirm-back"> but Weiwu
shied away from specifying that due to the likely change of design causing an upgrade path issue.
window.onConfirm is called if user hit "confirm";
window.close() causes the back button to be pressed.
-->
<ts:label>
<ts:string xml:lang="en">Enter</ts:string>
<ts:string xml:lang="zh">入場</ts:string>
<ts:string xml:lang="es">Entrar</ts:string>
</ts:label>
<ts:view xmlns="http://www.w3.org/1999/xhtml" xml:lang="en">
<style type="text/css">.ts-count {
font-family: "SourceSansPro";
font-weight: bolder;
font-size: 21px;
color: rgb(117, 185, 67);
}
.ts-category {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 21px;
color: rgb(67, 67, 67);
}
.ts-venue {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 16px;
color: rgb(67, 67, 67);
}
.ts-date {
font-family: "SourceSansPro";
font-weight: bold;
font-size: 14px;
color: rgb(112, 112, 112);
margin-left: 7px;
margin-right: 7px;
}
.ts-time {
font-family: "SourceSansPro";
font-weight: lighter;
font-size: 16px;
color: rgb(112, 112, 112);
}
html {
}
body {
padding: 0px;
margin: 0px;
}
div {
margin: 0px;
padding: 0px;
}
.data-icon {
height:16px;
vertical-align: middle
}
.tbml-count { font-family: "SourceSansPro"; font-weight: bolder; font-size: 21px; color: rgb(117, 185, 67);}.tbml-category { font-family: "SourceSansPro"; font-weight: lighter; font-size: 21px; color: rgb(67, 67, 67);}.tbml-venue { font-family: "SourceSansPro"; font-weight: lighter; font-size: 16px; color: rgb(67, 67, 67);}.tbml-date { font-family: "SourceSansPro"; font-weight: bold; font-size: 14px; color: rgb(112, 112, 112); margin-left: 7px; margin-right: 7px;}.tbml-time { font-family: "SourceSansPro"; font-weight: lighter; font-size: 16px; color: rgb(112, 112, 112);} html { } body { padding: 0px; margin: 0px; } div { margin: 0px; padding: 0px; } .data-icon { height:16px; vertical-align: middle }
</style>
<script type="text/javascript">//
class Token {
constructor(tokenInstance) {
this.props = tokenInstance
document.getElementById("contractAddress").value = this.props.EntryToken;
}
}
web3.tokens.dataChanged = (oldTokens, updatedTokens, tokenCardId) => {
const currentTokenInstance = updatedTokens.currentInstance;
document.getElementById(tokenCardId).innerHTML = new Token(currentTokenInstance).render();
};
document.addEventListener("DOMContentLoaded", function() {
window.onload = function startup() {
// 1. call API to fetch challenge
fetch('http://stormbird.duckdns.org:8080/api/getChallenge')
.then(function (response) {
return response.text()
})
.then(function (response) {
document.getElementById('msg').innerHTML = 'Challenge: ' + response
window.challenge = response
})
}
window.onConfirm = function onConfirm(signature) {
if (window.challenge === undefined || window.challenge.length == 0) return
const challenge = window.challenge
document.getElementById('status').innerHTML = 'Wait for signature...'
// 2. sign challenge to generate response
web3.personal.sign({ data: challenge }, function (error, value) {
if (error != null) {
document.getElementById('status').innerHTML = error
window.onload();
return
}
document.getElementById('status').innerHTML = 'Verifying credentials ...'
// 3. open door
let contractAddress = document.getElementById("contractAddress").textContent;
fetch(`http://stormbird.duckdns.org:8080/api/checkSignature?contract=${contractAddress}&challenge=${challenge}&sig=${value}`)
.then(function (response) {
return response.text()
})
.then(function (response) {
if (response == "pass") {
document.getElementById('status').innerHTML = 'Entrance granted!'
window.close()
} else {
document.getElementById('status').innerHTML = 'Failed with: ' + response
}
})
});
window.challenge = '';
document.getElementById('msg').innerHTML = '';
}
});
//
</script>
<body><h3>Welcome to Craig Wright's house!</h3>
<div id="msg">Preparing to unlock the entrance door.</div>
<div id="contractAddress"></div>
<div id="status"></div>
</body>
</ts:view>
</ts:card>
</ts:cards>
<ts:attribute name="tokenId" distinct="true">
<ts:type>
<ts:syntax>1.3.6.1.4.1.1466.115.121.1.40</ts:syntax>
</ts:type>
<ts:origins>
<ethereum:call function="balanceOf" contract="EntryToken">
<ts:data>
<ts:uint256 ref="ownerAddress"></ts:uint256>
</ts:data>
</ethereum:call>
</ts:origins>
</ts:attribute>
<ts:attribute name="locality">
<ts:type><ts:syntax>1.3.6.1.4.1.1466.115.121.1.15</ts:syntax></ts:type>
<ts:origins>
<ethereum:call as="utf8" contract="EntryToken" function="getLocality">
<ts:data>
<ts:uint256 ref="tokenId"></ts:uint256>
</ts:data>
</ethereum:call>
</ts:origins>
</ts:attribute>
<ts:attribute name="time">
<ts:type><ts:syntax>1.3.6.1.4.1.1466.115.121.1.24</ts:syntax></ts:type>
<ts:label>
<ts:string xml:lang="en">Time</ts:string>
<ts:string xml:lang="zh">时间</ts:string>
</ts:label>
<ts:origins>
<ts:token-id as="utf8" bitmask="
0000FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF0000000000000000000000
474d542b33000000000000000000000001075b2282f05255534b534101050002
"></ts:token-id>
</ts:origins>
</ts:attribute>
<ts:attribute name="expired"> <!-- boolean -->
<ts:type><ts:syntax>1.3.6.1.4.1.1466.115.121.1.7</ts:syntax></ts:type>
<ts:origins>
<ethereum:call as="bool" contract="EntryToken" function="isExpired">
<ts:data>
<ts:uint256 ref="tokenId"></ts:uint256>
</ts:data>
</ethereum:call>
</ts:origins>
</ts:attribute>
<ts:attribute name="street"> <!-- string -->
<ts:type><ts:syntax>1.3.6.1.4.1.1466.115.121.1.15</ts:syntax></ts:type>
<ts:origins>
<ethereum:call as="utf8" contract="EntryToken" function="getStreet">
<ts:data>
<ts:uint256 ref="tokenId"></ts:uint256>
</ts:data>
</ethereum:call>
</ts:origins>
</ts:attribute>
<ts:attribute name="building"> <!-- string -->
<ts:type><ts:syntax>1.3.6.1.4.1.1466.115.121.1.15</ts:syntax></ts:type>
<ts:origins>
<ethereum:call as="utf8" contract="EntryToken" function="getBuildingName">
<ts:data>
<ts:uint256 ref="tokenId"></ts:uint256>
</ts:data>
</ethereum:call>
</ts:origins>
</ts:attribute>
<ts:attribute name="state"> <!-- string -->
<ts:type><ts:syntax>1.3.6.1.4.1.1466.115.121.1.15</ts:syntax></ts:type>
<ts:origins>
<ethereum:call as="utf8" contract="EntryToken" function="getState">
<ts:data>
<ts:uint256 ref="tokenId"></ts:uint256>
</ts:data>
</ethereum:call>
</ts:origins>
</ts:attribute>
</ts:token>
OnConfirm:
@objc func proceed() {
let javaScriptToCallConfirm = """
if (window.onConfirm != null) {
onConfirm()
}
"""
tokenScriptRendererView.inject(javaScript: javaScriptToCallConfirm)
let userEntryIds = action.attributes.values.compactMap { $0.userEntryId }
let fetchUserEntries = userEntryIds
.map { "document.getElementById(\"\($0)\").value" }
.compactMap { tokenScriptRendererView.inject(javaScript: $0) }
guard let navigationController = navigationController else { return }
TokenScript.performTokenScriptAction(action, token: token, tokenId: tokenId, tokenHolder: tokenHolder, userEntryIds: userEntryIds, fetchUserEntries: fetchUserEntries, localRefsSource: tokenScriptRendererView, assetDefinitionStore: assetDefinitionStore, keystore: keystore, server: server, session: session, confirmTokenScriptActionTransactionDelegate: self, navigationController: navigationController)
}
https://github.com/AlphaWallet/alpha-wallet-ios/blob/07c8ad7f6770df647525504b7d955e0f85a56b55/AlphaWallet/Tokens/ViewControllers/TokenInstanceActionViewController.swift#L154
https://github.com/TokenScript/EIP5169TokenFactory/