feat: injecting scripts needed for communicating from browser to nim
- Changed WebView to WebEngineView - Created a new controller/view for the web3 provider - Created a private profile - Created a channel for comms browser - qml
This commit is contained in:
parent
e985e99f36
commit
c2567232b1
|
@ -0,0 +1,25 @@
|
||||||
|
import NimQml, chronicles
|
||||||
|
import ../../status/signals/types
|
||||||
|
import ../../status/status
|
||||||
|
import view
|
||||||
|
|
||||||
|
logScope:
|
||||||
|
topics = "web3-provider"
|
||||||
|
|
||||||
|
type Web3ProviderController* = ref object
|
||||||
|
status*: Status
|
||||||
|
view*: Web3ProviderView
|
||||||
|
variant*: QVariant
|
||||||
|
|
||||||
|
proc newController*(status: Status): Web3ProviderController =
|
||||||
|
result = Web3ProviderController()
|
||||||
|
result.status = status
|
||||||
|
result.view = newWeb3ProviderView(status)
|
||||||
|
result.variant = newQVariant(result.view)
|
||||||
|
|
||||||
|
proc delete*(self: Web3ProviderController) =
|
||||||
|
delete self.variant
|
||||||
|
delete self.view
|
||||||
|
|
||||||
|
proc init*(self: Web3ProviderController) =
|
||||||
|
discard
|
|
@ -0,0 +1,24 @@
|
||||||
|
import NimQml
|
||||||
|
import ../../status/status
|
||||||
|
|
||||||
|
QtObject:
|
||||||
|
type Web3ProviderView* = ref object of QObject
|
||||||
|
status*: Status
|
||||||
|
|
||||||
|
proc setup(self: Web3ProviderView) =
|
||||||
|
self.QObject.setup
|
||||||
|
|
||||||
|
proc delete*(self: Web3ProviderView) =
|
||||||
|
self.QObject.delete
|
||||||
|
|
||||||
|
proc newWeb3ProviderView*(status: Status): Web3ProviderView =
|
||||||
|
new(result, delete)
|
||||||
|
result = Web3ProviderView()
|
||||||
|
result.status = status
|
||||||
|
result.setup
|
||||||
|
|
||||||
|
proc postMessage*(self: Web3ProviderView, data: string): string {.slot.} =
|
||||||
|
# TODO: implement code from status-react/src/status_im/browser/core.cljs
|
||||||
|
echo "==========================================="
|
||||||
|
echo "Message received from JS web3 provider: ", data
|
||||||
|
return "Hello World!" # This can only be seen in chrome devtools
|
|
@ -7,6 +7,7 @@ import app/utilsView/core as utilsView
|
||||||
import app/profile/core as profile
|
import app/profile/core as profile
|
||||||
import app/onboarding/core as onboarding
|
import app/onboarding/core as onboarding
|
||||||
import app/login/core as login
|
import app/login/core as login
|
||||||
|
import app/provider/core as provider
|
||||||
import status/signals/core as signals
|
import status/signals/core as signals
|
||||||
import status/libstatus/types
|
import status/libstatus/types
|
||||||
import nim_status
|
import nim_status
|
||||||
|
@ -81,6 +82,9 @@ proc mainProc() =
|
||||||
var profile = profile.newController(status, changeLanguage)
|
var profile = profile.newController(status, changeLanguage)
|
||||||
engine.setRootContextProperty("profileModel", profile.variant)
|
engine.setRootContextProperty("profileModel", profile.variant)
|
||||||
|
|
||||||
|
var provider = provider.newController(status)
|
||||||
|
engine.setRootContextProperty("web3Provider", provider.variant)
|
||||||
|
|
||||||
var login = login.newController(status)
|
var login = login.newController(status)
|
||||||
var onboarding = onboarding.newController(status)
|
var onboarding = onboarding.newController(status)
|
||||||
|
|
||||||
|
@ -108,6 +112,7 @@ proc mainProc() =
|
||||||
|
|
||||||
defer:
|
defer:
|
||||||
error "TODO: if user is logged in, logout"
|
error "TODO: if user is logged in, logout"
|
||||||
|
provider.delete()
|
||||||
engine.delete()
|
engine.delete()
|
||||||
app.delete()
|
app.delete()
|
||||||
signalController.delete()
|
signalController.delete()
|
||||||
|
|
|
@ -1,25 +1,64 @@
|
||||||
import QtQuick 2.13
|
import QtQuick 2.13
|
||||||
import QtQuick.Layouts 1.13
|
import QtQuick.Layouts 1.13
|
||||||
import QtWebView 1.14
|
import QtWebEngine 1.10
|
||||||
|
import QtWebChannel 1.13
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
id: browserView
|
id: browserView
|
||||||
x: 0
|
|
||||||
y: 0
|
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
WebView {
|
// TODO: example qml webbrowser available here:
|
||||||
|
// https://doc.qt.io/qt-5/qtwebengine-webengine-quicknanobrowser-example.html
|
||||||
|
|
||||||
|
WebEngineProfile {
|
||||||
|
id: webProfile
|
||||||
|
offTheRecord: true // Private Mode on
|
||||||
|
persistentCookiesPolicy: WebEngineProfile.NoPersistentCookies
|
||||||
|
userScripts: [
|
||||||
|
WebEngineScript {
|
||||||
|
injectionPoint: WebEngineScript.DocumentCreation
|
||||||
|
sourceUrl: Qt.resolvedUrl("provider.js")
|
||||||
|
worldId: WebEngineScript.MainWorld // TODO: check https://doc.qt.io/qt-5/qml-qtwebengine-webenginescript.html#worldId-prop
|
||||||
|
},
|
||||||
|
WebEngineScript {
|
||||||
|
injectionPoint: WebEngineScript.DocumentCreation
|
||||||
|
name: "QWebChannel"
|
||||||
|
sourceUrl: "qrc:///qtwebchannel/qwebchannel.js"
|
||||||
|
worldId: WebEngineScript.MainWorld // TODO: check https://doc.qt.io/qt-5/qml-qtwebengine-webenginescript.html#worldId-prop
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
QtObject {
|
||||||
|
id: provider
|
||||||
|
WebChannel.id: "backend"
|
||||||
|
|
||||||
|
signal web3Response(string data);
|
||||||
|
|
||||||
|
function postMessage(data){
|
||||||
|
console.log("Calling nim web3provider with: ", data);
|
||||||
|
var result = web3Provider.postMessage(data);
|
||||||
|
web3Response(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WebChannel {
|
||||||
|
id: channel
|
||||||
|
registeredObjects: [provider]
|
||||||
|
}
|
||||||
|
|
||||||
|
WebEngineView {
|
||||||
id: browserContainer
|
id: browserContainer
|
||||||
anchors.top: parent.top
|
anchors.fill: parent
|
||||||
anchors.topMargin: 0
|
profile: webProfile
|
||||||
anchors.bottom: parent.bottom
|
url: "https://status-im.github.io/dapp/"
|
||||||
anchors.bottomMargin: 0
|
webChannel: channel
|
||||||
anchors.right: parent.right
|
onNewViewRequested: function(request) {
|
||||||
anchors.rightMargin: 0
|
// TODO: rramos: tabs can be handled here. see: https://doc.qt.io/qt-5/qml-qtwebengine-webengineview.html#newViewRequested-signal
|
||||||
anchors.left: parent.left
|
// In the meantime, I'm opening the content in the same webengineview
|
||||||
anchors.leftMargin: 0
|
request.openIn(browserContainer)
|
||||||
url: "https://dap.ps/"
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,283 @@
|
||||||
|
(function(){
|
||||||
|
// Based on
|
||||||
|
// https://github.com/status-im/status-react/blob/f9fb4d6974138a276b0cdcc6e4ea1611063e70ca/resources/js/provider.js
|
||||||
|
|
||||||
|
if(typeof EthereumProvider === "undefined"){
|
||||||
|
var callbackId = 0;
|
||||||
|
var callbacks = {};
|
||||||
|
|
||||||
|
|
||||||
|
var backend;
|
||||||
|
window.onload = function(){
|
||||||
|
new QWebChannel(qt.webChannelTransport, function(channel) {
|
||||||
|
backend = channel.objects.backend;
|
||||||
|
backend.web3Response.connect(function(data) {
|
||||||
|
// This can only be seen in chrome devtools
|
||||||
|
console.log("Received response from nim provider!!!!:");
|
||||||
|
console.log(data);
|
||||||
|
// TODO: implement the code from ReactNativeWebView.onMessage (~line 95)
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var bridgeSend = function (data) {
|
||||||
|
backend.postMessage(JSON.stringify(data))
|
||||||
|
}
|
||||||
|
|
||||||
|
var history = window.history;
|
||||||
|
var pushState = history.pushState;
|
||||||
|
history.pushState = function(state) {
|
||||||
|
setTimeout(function () {
|
||||||
|
bridgeSend({
|
||||||
|
type: 'history-state-changed',
|
||||||
|
navState: { url: location.href, title: document.title }
|
||||||
|
});
|
||||||
|
}, 100);
|
||||||
|
return pushState.apply(history, arguments);
|
||||||
|
};
|
||||||
|
|
||||||
|
function sendAPIrequest(permission, params) {
|
||||||
|
var messageId = callbackId++;
|
||||||
|
var params = params || {};
|
||||||
|
|
||||||
|
bridgeSend({
|
||||||
|
type: 'api-request',
|
||||||
|
permission: permission,
|
||||||
|
messageId: messageId,
|
||||||
|
params: params
|
||||||
|
});
|
||||||
|
|
||||||
|
return new Promise(function (resolve, reject) {
|
||||||
|
params['resolve'] = resolve;
|
||||||
|
params['reject'] = reject;
|
||||||
|
callbacks[messageId] = params;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function qrCodeResponse(data, callback){
|
||||||
|
var result = data.data;
|
||||||
|
var regex = new RegExp(callback.regex);
|
||||||
|
if (!result) {
|
||||||
|
if (callback.reject) {
|
||||||
|
callback.reject(new Error("Cancelled"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (regex.test(result)) {
|
||||||
|
if (callback.resolve) {
|
||||||
|
callback.resolve(result);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (callback.reject) {
|
||||||
|
callback.reject(new Error("Doesn't match"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function Unauthorized() {
|
||||||
|
this.name = "Unauthorized";
|
||||||
|
this.id = 4100;
|
||||||
|
this.code = 4100;
|
||||||
|
this.message = "The requested method and/or account has not been authorized by the user.";
|
||||||
|
}
|
||||||
|
Unauthorized.prototype = Object.create(Error.prototype);
|
||||||
|
|
||||||
|
function UserRejectedRequest() {
|
||||||
|
this.name = "UserRejectedRequest";
|
||||||
|
this.id = 4001;
|
||||||
|
this.code = 4001;
|
||||||
|
this.message = "The user rejected the request.";
|
||||||
|
}
|
||||||
|
UserRejectedRequest.prototype = Object.create(Error.prototype);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
TODO:
|
||||||
|
ReactNativeWebView.onMessage = function (message)
|
||||||
|
{
|
||||||
|
data = JSON.parse(message);
|
||||||
|
var id = data.messageId;
|
||||||
|
var callback = callbacks[id];
|
||||||
|
|
||||||
|
if (callback) {
|
||||||
|
if (data.type === "api-response") {
|
||||||
|
if (data.permission == 'qr-code'){
|
||||||
|
qrCodeResponse(data, callback);
|
||||||
|
} else if (data.isAllowed) {
|
||||||
|
if (data.permission == 'web3') {
|
||||||
|
window.statusAppcurrentAccountAddress = data.data[0];
|
||||||
|
}
|
||||||
|
callback.resolve(data.data);
|
||||||
|
} else {
|
||||||
|
callback.reject(new UserRejectedRequest());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (data.type === "web3-send-async-callback")
|
||||||
|
{
|
||||||
|
if (callback.beta)
|
||||||
|
{
|
||||||
|
if (data.error)
|
||||||
|
{
|
||||||
|
if (data.error.code == 4100)
|
||||||
|
callback.reject(new Unauthorized());
|
||||||
|
else
|
||||||
|
callback.reject(data.error);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
callback.resolve(data.result.result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (callback.results)
|
||||||
|
{
|
||||||
|
callback.results.push(data.error || data.result);
|
||||||
|
if (callback.results.length == callback.num)
|
||||||
|
callback.callback(undefined, callback.results);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
callback.callback(data.error, data.result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
*/
|
||||||
|
|
||||||
|
function web3Response (payload, result){
|
||||||
|
return {id: payload.id,
|
||||||
|
jsonrpc: "2.0",
|
||||||
|
result: result};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getSyncResponse (payload) {
|
||||||
|
if (payload.method == "eth_accounts" && (typeof window.statusAppcurrentAccountAddress !== "undefined")) {
|
||||||
|
return web3Response(payload, [window.statusAppcurrentAccountAddress])
|
||||||
|
} else if (payload.method == "eth_coinbase" && (typeof window.statusAppcurrentAccountAddress !== "undefined")) {
|
||||||
|
return web3Response(payload, window.statusAppcurrentAccountAddress)
|
||||||
|
} else if (payload.method == "net_version" || payload.method == "eth_chainId"){
|
||||||
|
return web3Response(payload, window.statusAppNetworkId)
|
||||||
|
} else if (payload.method == "eth_uninstallFilter"){
|
||||||
|
return web3Response(payload, true);
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var StatusAPI = function () {};
|
||||||
|
|
||||||
|
StatusAPI.prototype.getContactCode = function () {
|
||||||
|
return sendAPIrequest('contact-code');
|
||||||
|
};
|
||||||
|
|
||||||
|
var EthereumProvider = function () {};
|
||||||
|
|
||||||
|
EthereumProvider.prototype.isStatus = true;
|
||||||
|
EthereumProvider.prototype.status = new StatusAPI();
|
||||||
|
EthereumProvider.prototype.isConnected = function () { return true; };
|
||||||
|
|
||||||
|
EthereumProvider.prototype.enable = function () {
|
||||||
|
return sendAPIrequest('web3');
|
||||||
|
};
|
||||||
|
|
||||||
|
EthereumProvider.prototype.scanQRCode = function (regex) {
|
||||||
|
return sendAPIrequest('qr-code', {regex: regex});
|
||||||
|
};
|
||||||
|
|
||||||
|
EthereumProvider.prototype.request = function (requestArguments)
|
||||||
|
{
|
||||||
|
if (!requestArguments) {
|
||||||
|
return new Error('Request is not valid.');
|
||||||
|
}
|
||||||
|
var method = requestArguments.method;
|
||||||
|
|
||||||
|
if (!method) {
|
||||||
|
return new Error('Request is not valid.');
|
||||||
|
}
|
||||||
|
|
||||||
|
//Support for legacy send method
|
||||||
|
if (typeof method !== 'string') {
|
||||||
|
return this.sendSync(method);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (method == 'eth_requestAccounts'){
|
||||||
|
return sendAPIrequest('web3');
|
||||||
|
}
|
||||||
|
|
||||||
|
var syncResponse = getSyncResponse({method: method});
|
||||||
|
if (syncResponse){
|
||||||
|
return new Promise(function (resolve, reject) {
|
||||||
|
resolve(syncResponse.result);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var messageId = callbackId++;
|
||||||
|
var payload = {id: messageId,
|
||||||
|
jsonrpc: "2.0",
|
||||||
|
method: method,
|
||||||
|
params: requestArguments.params};
|
||||||
|
|
||||||
|
bridgeSend({type: 'web3-send-async-read-only',
|
||||||
|
messageId: messageId,
|
||||||
|
payload: payload});
|
||||||
|
|
||||||
|
return new Promise(function (resolve, reject) {
|
||||||
|
callbacks[messageId] = {beta: true,
|
||||||
|
resolve: resolve,
|
||||||
|
reject: reject};
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// (DEPRECATED) Support for legacy send method
|
||||||
|
EthereumProvider.prototype.send = function (method, params = [])
|
||||||
|
{
|
||||||
|
return this.request({method: method, params: params});
|
||||||
|
}
|
||||||
|
|
||||||
|
// (DEPRECATED) Support for legacy sendSync method
|
||||||
|
EthereumProvider.prototype.sendSync = function (payload)
|
||||||
|
{
|
||||||
|
if (payload.method == "eth_uninstallFilter"){
|
||||||
|
this.sendAsync(payload, function (res, err) {})
|
||||||
|
}
|
||||||
|
var syncResponse = getSyncResponse(payload);
|
||||||
|
if (syncResponse){
|
||||||
|
return syncResponse;
|
||||||
|
} else {
|
||||||
|
return web3Response(payload, null);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// (DEPRECATED) Support for legacy sendAsync method
|
||||||
|
EthereumProvider.prototype.sendAsync = function (payload, callback)
|
||||||
|
{
|
||||||
|
var syncResponse = getSyncResponse(payload);
|
||||||
|
if (syncResponse && callback) {
|
||||||
|
callback(null, syncResponse);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var messageId = callbackId++;
|
||||||
|
|
||||||
|
if (Array.isArray(payload))
|
||||||
|
{
|
||||||
|
callbacks[messageId] = {num: payload.length,
|
||||||
|
results: [],
|
||||||
|
callback: callback};
|
||||||
|
for (var i in payload) {
|
||||||
|
bridgeSend({type: 'web3-send-async-read-only',
|
||||||
|
messageId: messageId,
|
||||||
|
payload: payload[i]});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
callbacks[messageId] = {callback: callback};
|
||||||
|
bridgeSend({type: 'web3-send-async-read-only',
|
||||||
|
messageId: messageId,
|
||||||
|
payload: payload});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
window.ethereum = new EthereumProvider();
|
||||||
|
})();
|
|
@ -1,4 +1,5 @@
|
||||||
QT += quick
|
QT += quick
|
||||||
|
QT += webchannel
|
||||||
|
|
||||||
# The following define makes your compiler emit warnings if you use
|
# The following define makes your compiler emit warnings if you use
|
||||||
# any Qt feature that has been marked deprecated (the exact warnings
|
# any Qt feature that has been marked deprecated (the exact warnings
|
||||||
|
|
Loading…
Reference in New Issue