181 lines
3.8 KiB
Go
181 lines
3.8 KiB
Go
// Copyright 2015 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// +build darwin
|
|
// +build arm arm64
|
|
|
|
package app
|
|
|
|
/*
|
|
#cgo CFLAGS: -x objective-c
|
|
#cgo LDFLAGS: -framework Foundation -framework UIKit -framework GLKit -framework OpenGLES -framework QuartzCore
|
|
#include <sys/utsname.h>
|
|
#include <stdint.h>
|
|
#include <pthread.h>
|
|
|
|
extern struct utsname sysInfo;
|
|
|
|
void runApp(void);
|
|
void setContext(void* context);
|
|
uint64_t threadID();
|
|
*/
|
|
import "C"
|
|
import (
|
|
"log"
|
|
"runtime"
|
|
"strings"
|
|
"sync"
|
|
"unsafe"
|
|
|
|
"golang.org/x/mobile/event/config"
|
|
"golang.org/x/mobile/event/lifecycle"
|
|
"golang.org/x/mobile/event/paint"
|
|
"golang.org/x/mobile/event/touch"
|
|
"golang.org/x/mobile/geom"
|
|
"golang.org/x/mobile/gl"
|
|
)
|
|
|
|
var initThreadID uint64
|
|
|
|
func init() {
|
|
// Lock the goroutine responsible for initialization to an OS thread.
|
|
// This means the goroutine running main (and calling the run function
|
|
// below) is locked to the OS thread that started the program. This is
|
|
// necessary for the correct delivery of UIKit events to the process.
|
|
//
|
|
// A discussion on this topic:
|
|
// https://groups.google.com/forum/#!msg/golang-nuts/IiWZ2hUuLDA/SNKYYZBelsYJ
|
|
runtime.LockOSThread()
|
|
initThreadID = uint64(C.threadID())
|
|
}
|
|
|
|
func main(f func(App)) {
|
|
if tid := uint64(C.threadID()); tid != initThreadID {
|
|
log.Fatalf("app.Run called on thread %d, but app.init ran on %d", tid, initThreadID)
|
|
}
|
|
|
|
go func() {
|
|
f(app{})
|
|
// TODO(crawshaw): trigger runApp to return
|
|
}()
|
|
C.runApp()
|
|
panic("unexpected return from app.runApp")
|
|
}
|
|
|
|
var screenScale int // [UIScreen mainScreen].scale, either 1, 2, or 3.
|
|
|
|
//export setScreen
|
|
func setScreen(scale int) {
|
|
C.uname(&C.sysInfo)
|
|
name := C.GoString(&C.sysInfo.machine[0])
|
|
|
|
var v float32
|
|
|
|
switch {
|
|
case strings.HasPrefix(name, "iPhone"):
|
|
v = 163
|
|
case strings.HasPrefix(name, "iPad"):
|
|
// TODO: is there a better way to distinguish the iPad Mini?
|
|
switch name {
|
|
case "iPad2,5", "iPad2,6", "iPad2,7", "iPad4,4", "iPad4,5", "iPad4,6", "iPad4,7":
|
|
v = 163 // iPad Mini
|
|
default:
|
|
v = 132
|
|
}
|
|
default:
|
|
v = 163 // names like i386 and x86_64 are the simulator
|
|
}
|
|
|
|
if v == 0 {
|
|
log.Printf("unknown machine: %s", name)
|
|
v = 163 // emergency fallback
|
|
}
|
|
|
|
pixelsPerPt = v * float32(scale) / 72
|
|
screenScale = scale
|
|
}
|
|
|
|
//export updateConfig
|
|
func updateConfig(width, height int) {
|
|
widthPx := screenScale * width
|
|
heightPx := screenScale * height
|
|
eventsIn <- config.Event{
|
|
WidthPx: widthPx,
|
|
HeightPx: heightPx,
|
|
WidthPt: geom.Pt(float32(widthPx) / pixelsPerPt),
|
|
HeightPt: geom.Pt(float32(heightPx) / pixelsPerPt),
|
|
PixelsPerPt: pixelsPerPt,
|
|
}
|
|
}
|
|
|
|
var startedgl = false
|
|
|
|
// touchIDs is the current active touches. The position in the array
|
|
// is the ID, the value is the UITouch* pointer value.
|
|
//
|
|
// It is widely reported that the iPhone can handle up to 5 simultaneous
|
|
// touch events, while the iPad can handle 11.
|
|
var touchIDs [11]uintptr
|
|
|
|
var touchEvents struct {
|
|
sync.Mutex
|
|
pending []touch.Event
|
|
}
|
|
|
|
//export sendTouch
|
|
func sendTouch(cTouch, cTouchType uintptr, x, y float32) {
|
|
id := -1
|
|
for i, val := range touchIDs {
|
|
if val == cTouch {
|
|
id = i
|
|
break
|
|
}
|
|
}
|
|
if id == -1 {
|
|
for i, val := range touchIDs {
|
|
if val == 0 {
|
|
touchIDs[i] = cTouch
|
|
id = i
|
|
break
|
|
}
|
|
}
|
|
if id == -1 {
|
|
panic("out of touchIDs")
|
|
}
|
|
}
|
|
|
|
t := touch.Type(cTouchType)
|
|
if t == touch.TypeEnd {
|
|
touchIDs[id] = 0
|
|
}
|
|
|
|
eventsIn <- touch.Event{
|
|
X: x,
|
|
Y: y,
|
|
Sequence: touch.Sequence(id),
|
|
Type: t,
|
|
}
|
|
}
|
|
|
|
//export drawgl
|
|
func drawgl(ctx uintptr) {
|
|
if !startedgl {
|
|
startedgl = true
|
|
C.setContext(unsafe.Pointer(ctx))
|
|
// TODO(crawshaw): not just on process start.
|
|
sendLifecycle(lifecycle.StageFocused)
|
|
}
|
|
|
|
eventsIn <- paint.Event{}
|
|
|
|
for {
|
|
select {
|
|
case <-gl.WorkAvailable:
|
|
gl.DoWork()
|
|
case <-endPaint:
|
|
return
|
|
}
|
|
}
|
|
}
|