Chome Web Store: https://chrome.google.com/webstore/detail/pgjlaaokfffcapdcakncnhpmigjlnpei
Your dapp can use MetriMask to get information about a user's account status (whether they are logged into MetriMask, their account address, and balance). MetriMask also enables your dapp to listen to a window event for any changes to the user's account status. Your dapp can also use metrimask to make callcontract and sendtocontract calls to the blockchain.
To use any of the above functionality, your dapp will first need to initiate a long-lived connection between MetriMask's content script and background script.
The code to do this is already in MetriMask, your dapp just needs to trigger the function by posting a window message.
window.postMessage({ message: { type: 'CONNECT_METRIMASK' }}, '*')
This will populate the window.metrimask
object in your webpage. The window.metrimask.account
values are automatically updated when a user logs in/out or the account balance changes.
// window.metrimask
{
rpcProvider: MetriMaskRPCProvider,
account: {
loggedIn: true,
name: "2",
network: "TestNet",
address: "mNNiiJsBnPUZu4NwNtPaK9zyxD5ghV1By8",
balance: 49.10998413
}
}
You will probably want to refresh your dapp webpage when MetriMask is installed or updated. This allows your dapp to rerun
window.postMessage({ message: { type: 'CONNECT_METRIMASK' }}, '*')
which would have previously failed to do anything while MetriMask was not yet installed.
When MetriMask is installed or updated it will send all existing tabs an event message. To have that event message refresh your dapp, add the following event listener.
function handleMetriMaskInstalledOrUpdated(event) {
if (event.data.message && event.data.message.type === 'METRIMASK_INSTALLED_OR_UPDATED') {
// Refresh the page
window.location.reload()
}
}
window.addEventListener('message', handleMetriMaskInstalledOrUpdated, false);
After connecting MetriMask to your dapp, you can use an event listener to get notified of any changes to the user's account status(logging in/out, change in account balance).
function handleMetriMaskAcctChanged(event) {
if (event.data.message && event.data.message.type === "METRIMASK_ACCOUNT_CHANGED") {
if (event.data.message.payload.error){
// handle error
}
console.log("account:", event.data.message.payload.account)
}
}
window.addEventListener('message', handleMetriMaskAcctChanged, false);
Note that window.metrimask.account
will still get updated even if you don't set up this event listener; your Dapp just won't be notified of the changes.
RPC calls can be directly made via MetriMaskProvider
which is available to any webpage that connects to MetriMask.
Make sure that window.metrimask.rpcProvider
is defined before using it.
// callcontract
const contractAddress = 'ade0d0851168114b103d1cd891506ba562b19522';
const data = '06fdde03';
window.metrimask.rpcProvider.rawCall(
'callcontract',
[contractAddress, data]
).then((res) => console.log(res));
// sendtocontract
const contractAddress = 'ade0d0851168114b103d1cd891506ba562b19522';
const data = 'd0821b0e0000000000000000000000000000000000000000000000000000000000000001';
const metrixAmt = 1; // optional. defaults to 0.
const gasLimit = 250000; // optional. default: 250000, max: 40000000
const gasPrice = 5000; // optional. Metrix price per gas unit, default: 5000, min: 5000 (satoshi)
window.metrimask.rpcProvider.rawCall(
'sendtocontract',
[contractAddress, data, metrixAmt, gasLimit, gasPrice],
);
// signmessage
const url = window.location.origin;
const message = "This is a signed message!"; // The message to sign with currently logged in account.
const usePrefix = false; // (Optional) boolean - Use prefix '\x15Metrix Signed Message:\n' to sign the message.
window.metrimask.rpcProvider.signMessage([url, message, usePrefix]).then((result) => {
console.log(result);
}
// verifymessage
const message = "This is a signed message!"; // Message to verify
const address = window.metrimask.account.address; // Address to verify against, this gets the currently logged in one.
const signedMessage = "ThisIsTheSignedMessageString"; // A signed message
const usePrefix = false; // (Optional) boolean - Was the prefix '\x15Metrix Signed Message:\n' use to sign the original message.
window.metrimask.rpcProvider.verifyMessage([message, address, signedMessage, usePrefix]).then((response) => {
console.log(response); // true or false
}
// Handle incoming messages
function handleMessage(message) {
if (message.data.target == 'metrimask-inpage') {
// result: object
// error: string
const { result, error } = message.data.message.payload;
if (error) {
if (error === 'Not logged in. Please log in to MetriMask first.') {
// Show an alert dialog that the user needs to login first
alert(error);
} else {
// Handle different error than not logged in...
}
return;
}
// Do something with the message result...
}
}
window.addEventListener('message', handleMessage, false);
You may also use our Mweb3 convenience library to make sendtocontract
or callcontract
calls. See the instructions in the Github repo here: https://github.com/TheLindaProjectInc/mweb3-js
You can connect MetriMask to regtest. You will need to set the following in your metrixcore-node.json
"metrix-explorer": {
"apiPrefix": "insight-api",
"routePrefix": "explorer",
...
},
"metrix-insight-api": {
"routePrefix": "insight-api",
...
}
yarn start
in the project folder to build the dev version and wait for it to be built- Open Chrome and load URL:
chrome://extensions
- Turn
Developer mode
on in the top right - At the top, click
Load Unpacked Extension
- Navigate to your
metrimask/dist
folder - Click
Select
. The extension should now be loaded - Click on the MetriMask logo in your Chrome extensions bar to open
First Time Flow
appSalt
is generated on a per-install basis- User enters
password
in Login page password
+appSalt
runs throughscrpyt
encryption for ~3 seconds to generatepasswordHash
- User creates or imports wallet
passwordHash
+ wallet'sprivateKey
runs throughscrypt
encryption for ~1 second to generateencryptedPrivateKey
- Account is saved in storage with
encryptedPrivateKey
Return User Flow
- User enters password in Login page
password
+appSalt
runs throughscrpyt
encryption for ~3 seconds to generatepasswordHash
- Existing account is fetched from storage
passwordHash
is used to decrypted theencryptedPrivateKey
. On successful decryption of the wallet, the password is validated.