andlabs-ui/main.go

133 lines
3.4 KiB
Go
Raw Normal View History

2015-12-11 20:03:38 -06:00
// 11 december 2015
package ui
import (
2015-12-11 21:48:25 -06:00
"runtime"
2015-12-11 20:03:38 -06:00
"errors"
2015-12-11 21:48:25 -06:00
"sync"
"unsafe"
2015-12-11 20:03:38 -06:00
)
2015-12-11 21:48:25 -06:00
// #include "ui.h"
// extern void doQueued(void *);
2015-12-12 11:01:31 -06:00
// extern int doOnShouldQuit(void *);
2015-12-11 21:48:25 -06:00
// /* I forgot how dumb cgo is... ./main.go:73: cannot use _Cgo_ptr(_Cfpvar_fp_doQueued) (type unsafe.Pointer) as type *[0]byte in argument to _Cfunc_uiQueueMain */
// /* I'm pretty sure this worked before... */
// static inline void realQueueMain(void *x)
// {
// uiQueueMain(doQueued, x);
// }
2015-12-12 11:01:31 -06:00
// static inline int realOnShouldQuit(void)
// {
// uiOnShouldQuit(doOnShouldQuit, NULL);
// }
2015-12-11 20:03:38 -06:00
import "C"
// Main initializes package ui, runs f to set up the program,
// and executes the GUI main loop. f should set up the program's
// initial state: open the main window, create controls, and set up
// events. It should then return, at which point Main will
// process events until Quit is called, at which point Main will return
// nil. If package ui fails to initialize, Main returns an appropriate
// error.
func Main(f func()) error {
errchan := make(chan error)
go start(errchan, f)
return <-errchan
}
func start(errchan chan error, f func()) {
2015-12-11 21:48:25 -06:00
runtime.LockOSThread()
ensureMainThread()
2015-12-11 21:49:29 -06:00
2015-12-11 21:48:25 -06:00
// TODO HEAP SAFETY
opts := C.uiInitOptions{}
estr := C.uiInit(&opts)
if estr != nil {
2015-12-11 20:03:38 -06:00
errchan <- errors.New(C.GoString(estr))
2015-12-11 21:48:25 -06:00
C.uiFreeInitError(estr)
2015-12-11 20:03:38 -06:00
return
}
2015-12-12 11:01:31 -06:00
// set up OnShouldQuit()
C.realOnShouldQuit()
2015-12-11 20:03:38 -06:00
QueueMain(f)
2015-12-11 21:48:25 -06:00
C.uiMain()
2015-12-11 20:03:38 -06:00
errchan <- nil
}
2015-12-12 12:37:36 -06:00
// Quit queues a return from Main. It does not exit the program.
// It also does not immediately cause Main to return; Main will
// return when it next can. Quit must be called from the GUI thread.
2015-12-11 20:03:38 -06:00
func Quit() {
2015-12-11 21:48:25 -06:00
C.uiQuit()
2015-12-11 20:03:38 -06:00
}
2015-12-11 21:48:25 -06:00
// These prevent the passing of Go functions into C land.
// TODO make an actual sparse list instead of this monotonic map thingy
var (
qmmap = make(map[uintptr]func())
qmcurrent = uintptr(0)
qmlock sync.Mutex
)
2015-12-11 20:03:38 -06:00
// QueueMain queues f to be executed on the GUI thread when
// next possible. It returns immediately; that is, it does not wait
// for the function to actually be executed. QueueMain is the only
// function that can be called from other goroutines, and its
// primary purpose is to allow communication between other
// goroutines and the GUI thread. Calling QueueMain after Quit
// has been called results in undefined behavior.
func QueueMain(f func()) {
2015-12-11 21:48:25 -06:00
qmlock.Lock()
defer qmlock.Unlock()
2015-12-21 13:26:09 -06:00
n := uintptr(0)
for {
n = qmcurrent
qmcurrent++
if qmmap[n] == nil {
break
}
}
2015-12-11 21:48:25 -06:00
qmmap[n] = f
C.realQueueMain(unsafe.Pointer(n))
2015-12-11 20:03:38 -06:00
}
2015-12-11 21:48:25 -06:00
//export doQueued
func doQueued(nn unsafe.Pointer) {
qmlock.Lock()
n := uintptr(nn)
f := qmmap[n]
delete(qmmap, n)
// allow uiQueueMain() to be called by a queued function
// TODO explicitly allow this in libui too
qmlock.Unlock()
2015-12-11 20:03:38 -06:00
f()
}
2015-12-12 11:01:31 -06:00
// no need to lock this; this API is only safe on the main thread
var shouldQuitFunc func() bool
// OnShouldQuit schedules f to be exeucted when the OS wants
// the program to quit or when a Quit menu item has been clicked.
// Only one function may be registered at a time. If the function
// returns true, Quit will be called. If the function returns false, or
// if OnShouldQuit is never called. Quit will not be called and the
// OS will be told that the program needs to continue running.
func OnShouldQuit(f func() bool) {
shouldQuitFunc = f
}
//export doOnShouldQuit
func doOnShouldQuit(unused unsafe.Pointer) C.int {
if shouldQuitFunc == nil {
return 0
}
2015-12-12 12:07:57 -06:00
return frombool(shouldQuitFunc())
2015-12-12 11:01:31 -06:00
}