forked from AllenDang/cimgui-go
-
Notifications
You must be signed in to change notification settings - Fork 0
/
backend.go
207 lines (178 loc) · 5.76 KB
/
backend.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
package imgui
// extern void loopCallback();
// extern void beforeRender();
// extern void afterRender();
// extern void afterCreateContext();
// extern void beforeDestoryContext();
// extern void dropCallback(void*, int, char**);
// extern void keyCallback(void*, int, int, int, int);
// extern void sizeCallback(void*, int, int);
import "C"
import (
"errors"
"fmt"
"image"
"unsafe"
)
type voidCallbackFunc func()
var currentBackend backendCExpose
// TODO: Maybe we should get rid of it?
var textureManager TextureManager
//export loopCallback
func loopCallback() {
if currentBackend != nil {
if f := currentBackend.loopFunc(); f != nil {
f()
}
}
}
//export beforeRender
func beforeRender() {
if currentBackend != nil {
if f := currentBackend.beforeRenderHook(); f != nil {
f()
}
}
}
//export afterRender
func afterRender() {
if currentBackend != nil {
if f := currentBackend.afterRenderHook(); f != nil {
f()
}
}
}
//export afterCreateContext
func afterCreateContext() {
if currentBackend != nil {
if f := currentBackend.afterCreateHook(); f != nil {
f()
}
}
}
//export beforeDestoryContext
func beforeDestoryContext() {
if currentBackend != nil {
if f := currentBackend.beforeDestroyHook(); f != nil {
f()
}
}
}
//export keyCallback
func keyCallback(_ unsafe.Pointer, key, scanCode, action, mods C.int) {
if currentBackend != nil {
if f := currentBackend.keyCallback(); f != nil {
f(int(key), int(scanCode), int(action), int(mods))
}
}
}
//export sizeCallback
func sizeCallback(_ unsafe.Pointer, w, h C.int) {
if currentBackend != nil {
if f := currentBackend.sizeCallback(); f != nil {
f(int(w), int(h))
}
}
}
type (
DropCallback func([]string)
KeyCallback func(key, scanCode, action, mods int)
SizeChangeCallback func(w, h int)
)
type WindowCloseCallback[BackendFlagsT ~int] func(b Backend[BackendFlagsT])
//export closeCallback
func closeCallback(wnd unsafe.Pointer) {
currentBackend.closeCallback()(wnd)
}
//export dropCallback
func dropCallback(wnd unsafe.Pointer, count C.int, names **C.char) {
namesSlice := make([]string, int(count))
for i := 0; i < int(count); i++ {
var x *C.char
p := (**C.char)(unsafe.Pointer(uintptr(unsafe.Pointer(names)) + uintptr(i)*unsafe.Sizeof(x)))
namesSlice[i] = C.GoString(*p)
}
currentBackend.dropCallback()(namesSlice)
}
// Backend is a special interface that implements all methods required
// to render imgui application.
type Backend[BackendFlagsT ~int] interface {
SetAfterCreateContextHook(func())
SetBeforeDestroyContextHook(func())
SetBeforeRenderHook(func())
SetAfterRenderHook(func())
SetBgColor(color Vec4)
Run(func())
Refresh()
SetWindowPos(x, y int)
GetWindowPos() (x, y int32)
SetWindowSize(width, height int)
SetWindowSizeLimits(minWidth, minHeight, maxWidth, maxHeight int)
SetWindowTitle(title string)
DisplaySize() (width, height int32)
SetShouldClose(bool)
ContentScale() (xScale, yScale float32)
SetTargetFPS(fps uint)
SetDropCallback(DropCallback)
SetCloseCallback(WindowCloseCallback[BackendFlagsT])
SetKeyCallback(KeyCallback)
SetSizeChangeCallback(SizeChangeCallback)
// SetWindowFlags selected hint to specified value.
// ATTENTION: This method is able to set only one flag per call.
SetWindowFlags(flag BackendFlagsT, value int)
SetIcons(icons ...image.Image)
SetSwapInterval(interval BackendFlagsT) error
SetCursorPos(x, y float64)
SetInputMode(mode BackendFlagsT, value BackendFlagsT)
CreateWindow(title string, width, height int)
TextureManager
}
// TextureManager is a part of Backend.
//
// Why I separate it? Current impl of local texture.go needs to store this somewhere, and I don't want
// to make Texture relate on BackendFlagsT.
type TextureManager interface {
CreateTexture(pixels unsafe.Pointer, width, Height int) TextureID
CreateTextureRgba(img *image.RGBA, width, height int) TextureID
DeleteTexture(id TextureID)
}
type backendCExpose interface {
// for C callbacks
// What happens here is a bit tricky:
// - user sets these callbacks via Set* methods of the backend
// - callbacks are stored in currentContext variable
// - functions below just returns that callbacks
// - on top of this file is a set of global function with names similar to these below
// - these functions are exported to C
// - backend implementation uses C references to Go callbacks to share them (again ;-) )
// into backend code.
// As you can see this is all to convert Go callback int C callback
afterCreateHook() func()
beforeRenderHook() func()
loopFunc() func()
afterRenderHook() func()
beforeDestroyHook() func()
dropCallback() DropCallback
closeCallback() func(window unsafe.Pointer)
keyCallback() KeyCallback
sizeCallback() SizeChangeCallback
}
var CExposerError error = errors.New("CreateBackend was unable to extract C API Exposer from your backend. This is an error if you want to use C-related backend.")
// CreateBackend should to be called before using any cimgui methods.
// This enables the following features:
// - C-related callbacks: if your backend implements C functions, this will export some methods
// from your backend to C.
// - Advanced Texture Loading stuff. GL texture loading is a little complex. texture.go has some high-level logic.
//
// Error:
// - Error could be CExposerError when your backend does not support C. THIS MAY STILL WORK CORRECTLY.
// Specificly, this may happen if you use some backend that is not part of cimgui-go. In such a case
// just ignore the error.
func CreateBackend[BackendFlagsT ~int](backend Backend[BackendFlagsT]) (sameBackend Backend[BackendFlagsT], err error) {
var ok bool
if currentBackend, ok = backend.(backendCExpose); !ok {
err = fmt.Errorf("%T does not implement %T: %w", backend, currentBackend, CExposerError)
}
textureManager = backend
return backend, err
}