bsc/mist/ui_lib.go

288 lines
6.1 KiB
Go
Raw Normal View History

2014-07-02 01:13:50 +03:00
package main
2014-02-25 11:54:15 +02:00
import (
2014-08-15 14:45:34 +03:00
"bytes"
"fmt"
2014-08-07 17:35:47 +03:00
"path"
2014-08-15 14:45:34 +03:00
"strconv"
"strings"
2014-08-07 17:35:47 +03:00
2014-02-25 11:54:15 +02:00
"github.com/ethereum/eth-go"
"github.com/ethereum/eth-go/ethchain"
2014-08-15 14:45:34 +03:00
"github.com/ethereum/eth-go/ethcrypto"
2014-08-15 14:16:07 +03:00
"github.com/ethereum/eth-go/ethpipe"
2014-08-20 11:00:02 +03:00
"github.com/ethereum/eth-go/ethstate"
2014-02-25 11:54:15 +02:00
"github.com/ethereum/eth-go/ethutil"
"github.com/ethereum/go-ethereum/javascript"
2014-08-15 14:27:43 +03:00
"gopkg.in/qml.v1"
2014-02-25 11:54:15 +02:00
)
2014-04-12 07:12:10 +03:00
type memAddr struct {
Num string
Value string
}
2014-02-25 11:54:15 +02:00
// UI Library that has some basic functionality exposed
type UiLib struct {
2014-08-15 14:16:07 +03:00
*ethpipe.JSPipe
2014-02-25 11:54:15 +02:00
engine *qml.Engine
eth *eth.Ethereum
connected bool
assetPath string
2014-04-11 19:50:31 +03:00
// The main application window
win *qml.Window
Db *Debugger
DbWindow *DebuggerWindow
jsEngine *javascript.JSRE
2014-08-20 11:00:02 +03:00
filterCallbacks map[int][]int
}
func NewUiLib(engine *qml.Engine, eth *eth.Ethereum, assetPath string) *UiLib {
2014-09-14 01:13:47 +03:00
return &UiLib{JSPipe: ethpipe.NewJSPipe(eth), engine: engine, eth: eth, assetPath: assetPath, jsEngine: javascript.NewJSRE(eth), filterCallbacks: make(map[int][]int)} //, filters: make(map[int]*ethpipe.JSFilter)}
}
2014-09-14 01:13:47 +03:00
func (self *UiLib) Notef(args []interface{}) {
logger.Infoln(args...)
2014-09-08 01:50:25 +03:00
}
2014-08-15 14:45:34 +03:00
func (self *UiLib) LookupDomain(domain string) string {
world := self.World()
if len(domain) > 32 {
domain = string(ethcrypto.Sha3Bin([]byte(domain)))
}
data := world.Config().Get("DnsReg").StorageString(domain).Bytes()
// Left padded = A record, Right padded = CNAME
2014-08-23 16:30:23 +03:00
if len(data) > 0 && data[0] == 0 {
2014-08-15 14:45:34 +03:00
data = bytes.TrimLeft(data, "\x00")
var ipSlice []string
for _, d := range data {
ipSlice = append(ipSlice, strconv.Itoa(int(d)))
}
return strings.Join(ipSlice, ".")
} else {
data = bytes.TrimRight(data, "\x00")
return string(data)
}
}
2014-09-17 16:58:44 +03:00
func (self *UiLib) PastPeers() *ethutil.List {
return ethutil.NewList(eth.PastPeers())
}
func (self *UiLib) ImportTx(rlpTx string) {
tx := ethchain.NewTransactionFromBytes(ethutil.Hex2Bytes(rlpTx))
self.eth.TxPool().QueueTransaction(tx)
}
func (self *UiLib) EvalJavascriptFile(path string) {
self.jsEngine.LoadExtFile(path[7:])
2014-02-25 11:54:15 +02:00
}
func (self *UiLib) EvalJavascriptString(str string) string {
value, err := self.jsEngine.Run(str)
if err != nil {
return err.Error()
}
return fmt.Sprintf("%v", value)
}
func (ui *UiLib) OpenQml(path string) {
container := NewQmlApplication(path[7:], ui)
app := NewExtApplication(container, ui)
2014-02-25 11:54:15 +02:00
go app.run()
2014-02-25 11:54:15 +02:00
}
func (ui *UiLib) OpenHtml(path string) {
container := NewHtmlApplication(path, ui)
app := NewExtApplication(container, ui)
go app.run()
}
2014-08-07 17:35:47 +03:00
func (ui *UiLib) OpenBrowser() {
ui.OpenHtml("file://" + ui.AssetPath("ext/home.html"))
}
2014-04-28 01:25:01 +03:00
func (ui *UiLib) Muted(content string) {
component, err := ui.engine.LoadFile(ui.AssetPath("qml/muted.qml"))
if err != nil {
logger.Debugln(err)
2014-04-28 01:25:01 +03:00
return
}
win := component.CreateWindow(nil)
go func() {
path := "file://" + ui.AssetPath("muted/index.html")
win.Set("url", path)
win.Show()
win.Wait()
}()
}
2014-02-28 13:18:19 +02:00
func (ui *UiLib) Connect(button qml.Object) {
2014-02-25 11:54:15 +02:00
if !ui.connected {
2014-05-09 15:51:02 +03:00
ui.eth.Start(true)
2014-02-28 13:18:19 +02:00
ui.connected = true
button.Set("enabled", false)
2014-02-25 11:54:15 +02:00
}
}
func (ui *UiLib) ConnectToPeer(addr string) {
ui.eth.ConnectToPeer(addr)
}
func (ui *UiLib) AssetPath(p string) string {
return path.Join(ui.assetPath, p)
}
func (self *UiLib) StartDbWithContractAndData(contractHash, data string) {
dbWindow := NewDebuggerWindow(self)
object := self.eth.StateManager().CurrentState().GetStateObject(ethutil.Hex2Bytes(contractHash))
if len(object.Code) > 0 {
dbWindow.SetCode("0x" + ethutil.Bytes2Hex(object.Code))
}
2014-05-28 19:11:27 +03:00
dbWindow.SetData("0x" + data)
dbWindow.Show()
}
func (self *UiLib) StartDbWithCode(code string) {
dbWindow := NewDebuggerWindow(self)
dbWindow.SetCode("0x" + code)
dbWindow.Show()
}
2014-05-27 11:42:37 +03:00
func (self *UiLib) StartDebugger() {
dbWindow := NewDebuggerWindow(self)
dbWindow.Show()
}
2014-08-20 11:00:02 +03:00
2014-09-14 01:13:47 +03:00
func (self *UiLib) NewFilter(object map[string]interface{}) int {
filter, id := self.eth.InstallFilter(object)
2014-08-20 11:00:02 +03:00
filter.MessageCallback = func(messages ethstate.Messages) {
2014-09-14 01:13:47 +03:00
self.win.Root().Call("invokeFilterCallback", ethpipe.ToJSMessages(messages), id)
2014-08-20 11:00:02 +03:00
}
2014-08-20 13:42:49 +03:00
2014-09-14 01:13:47 +03:00
return id
2014-08-20 13:42:49 +03:00
}
2014-09-14 01:13:47 +03:00
func (self *UiLib) NewFilterString(typ string) int {
filter, id := self.eth.InstallFilter(nil)
filter.BlockCallback = func(block *ethchain.Block) {
self.win.Root().Call("invokeFilterCallback", "{}", id)
2014-08-20 13:42:49 +03:00
}
2014-08-20 11:00:02 +03:00
2014-09-14 01:13:47 +03:00
return id
2014-08-20 11:00:02 +03:00
}
2014-09-14 01:13:47 +03:00
func (self *UiLib) Messages(id int) *ethutil.List {
filter := self.eth.GetFilter(id)
2014-08-20 11:00:02 +03:00
if filter != nil {
2014-09-14 01:13:47 +03:00
messages := filter.Find()
return ethpipe.ToJSMessages(messages)
2014-08-20 11:00:02 +03:00
}
2014-09-14 01:13:47 +03:00
return ethutil.EmptyList()
2014-08-20 11:00:02 +03:00
}
2014-09-14 01:13:47 +03:00
func (self *UiLib) UninstallFilter(id int) {
self.eth.UninstallFilter(id)
2014-08-20 11:00:02 +03:00
}
2014-08-20 13:42:49 +03:00
2014-09-22 15:54:27 +03:00
func mapToTxParams(object map[string]interface{}) map[string]string {
2014-08-20 13:42:49 +03:00
// Default values
if object["from"] == nil {
object["from"] = ""
}
if object["to"] == nil {
object["to"] = ""
}
if object["value"] == nil {
object["value"] = ""
}
if object["gas"] == nil {
object["gas"] = ""
}
if object["gasPrice"] == nil {
object["gasPrice"] = ""
}
var dataStr string
var data []string
if list, ok := object["data"].(*qml.List); ok {
list.Convert(&data)
2014-09-22 15:54:27 +03:00
} else if str, ok := object["data"].(string); ok {
data = []string{str}
2014-08-20 13:42:49 +03:00
}
for _, str := range data {
if ethutil.IsHex(str) {
str = str[2:]
if len(str) != 64 {
str = ethutil.LeftPadString(str, 64)
}
} else {
str = ethutil.Bytes2Hex(ethutil.LeftPadBytes(ethutil.Big(str).Bytes(), 32))
}
dataStr += str
}
2014-09-22 15:54:27 +03:00
object["data"] = dataStr
fmt.Println(object)
conv := make(map[string]string)
for key, value := range object {
if v, ok := value.(string); ok {
conv[key] = v
}
}
return conv
}
func (self *UiLib) Transact(params map[string]interface{}) (*ethpipe.JSReceipt, error) {
object := mapToTxParams(params)
2014-08-20 13:42:49 +03:00
return self.JSPipe.Transact(
2014-09-22 15:54:27 +03:00
object["from"],
object["to"],
object["value"],
object["gas"],
object["gasPrice"],
object["data"],
2014-08-20 13:42:49 +03:00
)
}
func (self *UiLib) Compile(code string) (string, error) {
bcode, err := ethutil.Compile(code, false)
if err != nil {
return err.Error(), err
}
return ethutil.Bytes2Hex(bcode), err
}
2014-09-22 15:54:27 +03:00
func (self *UiLib) Call(params map[string]interface{}) (string, error) {
object := mapToTxParams(params)
return self.JSPipe.Execute(
object["to"],
object["value"],
object["gas"],
object["gasPrice"],
object["data"],
)
}