go-ethereum/cmd/mist/ui_lib.go

301 lines
6.6 KiB
Go
Raw Normal View History

2015-01-06 13:13:57 +02:00
/*
This file is part of go-ethereum
2014-10-23 16:48:53 +03:00
2015-01-06 13:13:57 +02:00
go-ethereum is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
go-ethereum is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
*/
/**
* @authors
* Jeffrey Wilcke <i@jev.io>
*/
2014-07-02 01:13:50 +03:00
package main
2014-02-25 11:54:15 +02:00
import (
"fmt"
2015-01-28 15:02:36 +02:00
"io/ioutil"
2014-08-07 17:35:47 +03:00
"path"
2014-12-04 11:28:02 +02:00
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/eth"
2015-03-16 12:27:38 +02:00
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/event/filter"
"github.com/ethereum/go-ethereum/javascript"
2014-10-31 15:30:08 +02:00
"github.com/ethereum/go-ethereum/xeth"
2015-01-28 15:51:54 +02:00
"github.com/obscuren/qml"
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 {
2015-01-28 19:35:49 +02:00
*xeth.XEth
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
2015-03-10 13:06:16 +02:00
win *qml.Window
jsEngine *javascript.JSRE
2014-08-20 11:00:02 +03:00
filterCallbacks map[int][]int
filterManager *filter.FilterManager
}
func NewUiLib(engine *qml.Engine, eth *eth.Ethereum, assetPath string) *UiLib {
x := xeth.New(eth, nil)
lib := &UiLib{XEth: x, engine: engine, eth: eth, assetPath: assetPath, jsEngine: javascript.NewJSRE(x), filterCallbacks: make(map[int][]int)} //, filters: make(map[int]*xeth.JSFilter)}
lib.filterManager = filter.NewFilterManager(eth.EventMux())
go lib.filterManager.Start()
return lib
}
2014-09-14 01:13:47 +03:00
func (self *UiLib) Notef(args []interface{}) {
2014-10-31 13:56:05 +02:00
guilogger.Infoln(args...)
2014-09-08 01:50:25 +03:00
}
func (self *UiLib) ImportTx(rlpTx string) {
2015-03-16 12:27:38 +02:00
tx := types.NewTransactionFromBytes(common.Hex2Bytes(rlpTx))
err := self.eth.TxPool().Add(tx)
if err != nil {
guilogger.Infoln("import tx failed ", err)
}
}
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)
}
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 {
2014-10-31 13:56:05 +02:00
guilogger.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 {
ui.eth.Start()
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(nodeURL string) {
if err := ui.eth.SuggestPeer(nodeURL); err != nil {
guilogger.Infoln("SuggestPeer error: " + err.Error())
2015-01-05 18:12:52 +02:00
}
2014-02-25 11:54:15 +02:00
}
func (ui *UiLib) AssetPath(p string) string {
return path.Join(ui.assetPath, p)
}
2014-12-22 14:23:11 +02:00
func (self *UiLib) Transact(params map[string]interface{}) (string, error) {
object := mapToTxParams(params)
2015-01-28 19:35:49 +02:00
return self.XEth.Transact(
object["from"],
2014-12-22 14:23:11 +02:00
object["to"],
object["value"],
object["gas"],
object["gasPrice"],
object["data"],
)
}
func (self *UiLib) Compile(code string) (string, error) {
2015-03-16 12:27:38 +02:00
bcode, err := common.Compile(code, false)
2014-12-22 14:23:11 +02:00
if err != nil {
return err.Error(), err
}
2015-03-16 12:27:38 +02:00
return common.Bytes2Hex(bcode), err
2014-12-22 14:23:11 +02:00
}
func (self *UiLib) Call(params map[string]interface{}) (string, error) {
object := mapToTxParams(params)
2015-03-12 18:09:25 +02:00
return self.XEth.Call(
object["from"],
2014-12-22 14:23:11 +02:00
object["to"],
object["value"],
object["gas"],
object["gasPrice"],
object["data"],
)
}
func (self *UiLib) AddLocalTransaction(to, data, gas, gasPrice, value string) int {
2015-02-09 17:20:34 +02:00
return 0
/*
return self.miner.AddLocalTx(&miner.LocalTx{
2015-03-16 12:27:38 +02:00
To: common.Hex2Bytes(to),
Data: common.Hex2Bytes(data),
2015-02-09 17:20:34 +02:00
Gas: gas,
GasPrice: gasPrice,
Value: value,
}) - 1
*/
2014-12-22 14:23:11 +02:00
}
func (self *UiLib) RemoveLocalTransaction(id int) {
2015-02-09 17:20:34 +02:00
//self.miner.RemoveLocalTx(id)
2014-12-22 14:23:11 +02:00
}
func (self *UiLib) ToggleMining() bool {
if !self.eth.IsMining() {
err := self.eth.StartMining()
return err == nil
2014-12-22 14:23:11 +02:00
} else {
self.eth.StopMining()
2014-12-22 14:23:11 +02:00
return false
}
}
func (self *UiLib) ToHex(data string) string {
2015-03-16 12:27:38 +02:00
return "0x" + common.Bytes2Hex([]byte(data))
2014-12-22 14:23:11 +02:00
}
func (self *UiLib) ToAscii(data string) string {
start := 0
if len(data) > 1 && data[0:2] == "0x" {
start = 2
}
2015-03-16 12:27:38 +02:00
return string(common.Hex2Bytes(data[start:]))
2014-12-22 14:23:11 +02:00
}
/// Ethereum filter methods
func (self *UiLib) NewFilter(object map[string]interface{}, view *qml.Common) (id int) {
2015-01-28 15:02:36 +02:00
/* TODO remove me
filter := qt.NewFilterFromMap(object, self.eth)
2014-10-31 15:43:14 +02:00
filter.MessageCallback = func(messages state.Messages) {
2015-01-28 19:35:49 +02:00
view.Call("messages", xeth.ToMessages(messages), id)
2014-08-20 11:00:02 +03:00
}
id = self.filterManager.InstallFilter(filter)
2014-09-14 01:13:47 +03:00
return id
2015-01-28 15:02:36 +02:00
*/
return 0
2014-08-20 13:42:49 +03:00
}
func (self *UiLib) NewFilterString(typ string, view *qml.Common) (id int) {
2015-01-28 15:02:36 +02:00
/* TODO remove me
2014-12-04 11:28:02 +02:00
filter := core.NewFilter(self.eth)
filter.BlockCallback = func(block *types.Block) {
view.Call("messages", "{}", id)
2014-08-20 13:42:49 +03:00
}
id = self.filterManager.InstallFilter(filter)
2014-09-14 01:13:47 +03:00
return id
2015-01-28 15:02:36 +02:00
*/
return 0
2014-08-20 11:00:02 +03:00
}
2015-03-16 12:27:38 +02:00
func (self *UiLib) Messages(id int) *common.List {
2015-01-28 15:02:36 +02:00
/* TODO remove me
filter := self.filterManager.GetFilter(id)
2014-08-20 11:00:02 +03:00
if filter != nil {
2015-01-28 19:35:49 +02:00
messages := xeth.ToMessages(filter.Find())
2014-09-14 01:13:47 +03:00
2014-09-24 21:40:18 +03:00
return messages
2014-08-20 11:00:02 +03:00
}
2015-01-28 15:02:36 +02:00
*/
2014-09-14 01:13:47 +03:00
2015-03-16 12:27:38 +02:00
return common.EmptyList()
2014-08-20 11:00:02 +03:00
}
2015-01-28 15:02:36 +02:00
func (self *UiLib) ReadFile(p string) string {
content, err := ioutil.ReadFile(self.AssetPath(path.Join("ext", p)))
if err != nil {
guilogger.Infoln("error reading file", p, ":", err)
}
return string(content)
}
2014-09-14 01:13:47 +03:00
func (self *UiLib) UninstallFilter(id int) {
self.filterManager.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 {
2015-03-16 12:27:38 +02:00
if common.IsHex(str) {
2014-08-20 13:42:49 +03:00
str = str[2:]
if len(str) != 64 {
2015-03-16 12:27:38 +02:00
str = common.LeftPadString(str, 64)
2014-08-20 13:42:49 +03:00
}
} else {
2015-03-16 12:27:38 +02:00
str = common.Bytes2Hex(common.LeftPadBytes(common.Big(str).Bytes(), 32))
2014-08-20 13:42:49 +03:00
}
dataStr += str
}
2014-09-22 15:54:27 +03:00
object["data"] = dataStr
conv := make(map[string]string)
for key, value := range object {
if v, ok := value.(string); ok {
conv[key] = v
}
}
return conv
}