darwin_ios.go (5149B)
1 // Copyright 2015 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 //go:build darwin && ios 6 // +build darwin,ios 7 8 package app 9 10 /* 11 #cgo CFLAGS: -x objective-c -DGL_SILENCE_DEPRECATION 12 #cgo LDFLAGS: -framework Foundation -framework UIKit -framework GLKit -framework OpenGLES -framework QuartzCore 13 #include <sys/utsname.h> 14 #include <stdint.h> 15 #include <pthread.h> 16 #include <UIKit/UIDevice.h> 17 #import <GLKit/GLKit.h> 18 19 extern struct utsname sysInfo; 20 21 void runApp(void); 22 void makeCurrentContext(GLintptr ctx); 23 void swapBuffers(GLintptr ctx); 24 uint64_t threadID(); 25 */ 26 import "C" 27 import ( 28 "log" 29 "runtime" 30 "strings" 31 "sync" 32 33 "golang.org/x/mobile/event/lifecycle" 34 "golang.org/x/mobile/event/paint" 35 "golang.org/x/mobile/event/size" 36 "golang.org/x/mobile/event/touch" 37 "golang.org/x/mobile/geom" 38 ) 39 40 var initThreadID uint64 41 42 func init() { 43 // Lock the goroutine responsible for initialization to an OS thread. 44 // This means the goroutine running main (and calling the run function 45 // below) is locked to the OS thread that started the program. This is 46 // necessary for the correct delivery of UIKit events to the process. 47 // 48 // A discussion on this topic: 49 // https://groups.google.com/forum/#!msg/golang-nuts/IiWZ2hUuLDA/SNKYYZBelsYJ 50 runtime.LockOSThread() 51 initThreadID = uint64(C.threadID()) 52 } 53 54 func main(f func(App)) { 55 if tid := uint64(C.threadID()); tid != initThreadID { 56 log.Fatalf("app.Run called on thread %d, but app.init ran on %d", tid, initThreadID) 57 } 58 59 go func() { 60 f(theApp) 61 // TODO(crawshaw): trigger runApp to return 62 }() 63 C.runApp() 64 panic("unexpected return from app.runApp") 65 } 66 67 var pixelsPerPt float32 68 var screenScale int // [UIScreen mainScreen].scale, either 1, 2, or 3. 69 70 //export setScreen 71 func setScreen(scale int) { 72 C.uname(&C.sysInfo) 73 name := C.GoString(&C.sysInfo.machine[0]) 74 75 var v float32 76 77 switch { 78 case strings.HasPrefix(name, "iPhone"): 79 v = 163 80 case strings.HasPrefix(name, "iPad"): 81 // TODO: is there a better way to distinguish the iPad Mini? 82 switch name { 83 case "iPad2,5", "iPad2,6", "iPad2,7", "iPad4,4", "iPad4,5", "iPad4,6", "iPad4,7": 84 v = 163 // iPad Mini 85 default: 86 v = 132 87 } 88 default: 89 v = 163 // names like i386 and x86_64 are the simulator 90 } 91 92 if v == 0 { 93 log.Printf("unknown machine: %s", name) 94 v = 163 // emergency fallback 95 } 96 97 pixelsPerPt = v * float32(scale) / 72 98 screenScale = scale 99 } 100 101 //export updateConfig 102 func updateConfig(width, height, orientation int32) { 103 o := size.OrientationUnknown 104 switch orientation { 105 case C.UIDeviceOrientationPortrait, C.UIDeviceOrientationPortraitUpsideDown: 106 o = size.OrientationPortrait 107 case C.UIDeviceOrientationLandscapeLeft, C.UIDeviceOrientationLandscapeRight: 108 o = size.OrientationLandscape 109 } 110 widthPx := screenScale * int(width) 111 heightPx := screenScale * int(height) 112 theApp.eventsIn <- size.Event{ 113 WidthPx: widthPx, 114 HeightPx: heightPx, 115 WidthPt: geom.Pt(float32(widthPx) / pixelsPerPt), 116 HeightPt: geom.Pt(float32(heightPx) / pixelsPerPt), 117 PixelsPerPt: pixelsPerPt, 118 Orientation: o, 119 } 120 theApp.eventsIn <- paint.Event{External: true} 121 } 122 123 // touchIDs is the current active touches. The position in the array 124 // is the ID, the value is the UITouch* pointer value. 125 // 126 // It is widely reported that the iPhone can handle up to 5 simultaneous 127 // touch events, while the iPad can handle 11. 128 var touchIDs [11]uintptr 129 130 var touchEvents struct { 131 sync.Mutex 132 pending []touch.Event 133 } 134 135 //export sendTouch 136 func sendTouch(cTouch, cTouchType uintptr, x, y float32) { 137 id := -1 138 for i, val := range touchIDs { 139 if val == cTouch { 140 id = i 141 break 142 } 143 } 144 if id == -1 { 145 for i, val := range touchIDs { 146 if val == 0 { 147 touchIDs[i] = cTouch 148 id = i 149 break 150 } 151 } 152 if id == -1 { 153 panic("out of touchIDs") 154 } 155 } 156 157 t := touch.Type(cTouchType) 158 if t == touch.TypeEnd { 159 touchIDs[id] = 0 160 } 161 162 theApp.eventsIn <- touch.Event{ 163 X: x, 164 Y: y, 165 Sequence: touch.Sequence(id), 166 Type: t, 167 } 168 } 169 170 //export lifecycleDead 171 func lifecycleDead() { theApp.sendLifecycle(lifecycle.StageDead) } 172 173 //export lifecycleAlive 174 func lifecycleAlive() { theApp.sendLifecycle(lifecycle.StageAlive) } 175 176 //export lifecycleVisible 177 func lifecycleVisible() { theApp.sendLifecycle(lifecycle.StageVisible) } 178 179 //export lifecycleFocused 180 func lifecycleFocused() { theApp.sendLifecycle(lifecycle.StageFocused) } 181 182 //export startloop 183 func startloop(ctx C.GLintptr) { 184 go theApp.loop(ctx) 185 } 186 187 // loop is the primary drawing loop. 188 // 189 // After UIKit has captured the initial OS thread for processing UIKit 190 // events in runApp, it starts loop on another goroutine. It is locked 191 // to an OS thread for its OpenGL context. 192 func (a *app) loop(ctx C.GLintptr) { 193 runtime.LockOSThread() 194 C.makeCurrentContext(ctx) 195 196 workAvailable := a.worker.WorkAvailable() 197 198 for { 199 select { 200 case <-workAvailable: 201 a.worker.DoWork() 202 case <-theApp.publish: 203 loop1: 204 for { 205 select { 206 case <-workAvailable: 207 a.worker.DoWork() 208 default: 209 break loop1 210 } 211 } 212 C.swapBuffers(ctx) 213 theApp.publishResult <- PublishResult{} 214 } 215 } 216 }