token script study

https://www.tokenscript.org/

https://community.tokenscript.org/

https://medium.com/alphawallet/tokenscript/home

TokenScript

https://github.com/TokenScript/TokenScript/tree https://github.com/TokenScript/TokenScriptTestContracts/tree/master https://github.com/TokenScript/TokenScript-Examples/tree/master/tutorial

example: EntryToken

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 `&lt;div&gt;
            &lt;div&gt;
                &lt;span class="ts-count"&gt;x${this.props._count}&lt;/span&gt;  &lt;span class="ts-category"&gt;${this.props.label}&lt;/span&gt;
            &lt;/div&gt;
            &lt;div&gt;
                &lt;span class="ts-venue"&gt;${this.props.building}&lt;/span&gt;
            &lt;/div&gt;
                &lt;div style="margin: 0px; padding:0px; clear: both; height: 6px"&gt;
                &amp;nbsp;
                &lt;/div&gt;
            &lt;div&gt;
                &lt;img src="" class="data-icon"/&gt;
                &lt;span class="ts-date"&gt;${date}&lt;/span&gt;
            &lt;/div&gt;
            &lt;div&gt;
                &lt;span class="ts-time"&gt;${time}, ${this.props.locality}&lt;/span&gt;
            &lt;/div&gt;
            &lt;/div&gt;`;
    }
}

web3.tokens.dataChanged = (oldTokens, updatedTokens, tokenCardId) =&gt; {
    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) =&gt; {
    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}&amp;challenge=${challenge}&amp;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

EIP5169 powered Token with script and firmware code

https://github.com/TokenScript/EIP5169TokenFactory/

Web3E