-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgohFunctions.go
executable file
·393 lines (348 loc) · 12.3 KB
/
gohFunctions.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
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
// gohFunctions.go
/*
Source file auto-generated on Fri, 02 Apr 2021 13:10:55 using Gotk3 Objects Handler v1.7.5 ©2018-21 hfmrow
This software use gotk3 that is licensed under the ISC License:
https://github.com/gotk3/gotk3/blob/master/LICENSE
Copyright ©2018-21 H.F.M - Search Engine v1.9 github.com/hfmrow/search-engine
This program comes with absolutely no warranty. See the The MIT License (MIT) for details:
https://opensource.org/licenses/mit-license.php
*/
package main
import (
"bytes"
"compress/gzip"
"fmt"
"io"
"io/ioutil"
"log"
"os"
"os/user"
"path"
"path/filepath"
"reflect"
"regexp"
"strings"
"github.com/gotk3/gotk3/gdk"
"github.com/gotk3/gotk3/glib"
"github.com/gotk3/gotk3/gtk"
)
/*******************************************************/
/* Functions declarations, used to initialize objects */
/*****************************************************/
// newBuilder: initialise builder with glade xml string
func newBuilder(varPath interface{}) (err error) {
var Gtk3Interface []byte
if Gtk3Interface, err = getBytesFromVarAsset(varPath); err == nil {
if mainObjects.mainUiBuilder, err = gtk.BuilderNew(); err == nil {
err = mainObjects.mainUiBuilder.AddFromString(string(Gtk3Interface))
}
}
return err
}
// loadObject: Load GtkObject to be transtyped ...
func loadObject(name string) (newObj glib.IObject) {
var err error
if newObj, err = mainObjects.mainUiBuilder.GetObject(name); err != nil {
fmt.Printf("Unable to load %s object, maybe it was deleted from the Glade file ... : %s\n%s\n",
name, err.Error(),
fmt.Sprint("An update with GOH may avoid this issue."))
os.Exit(1)
}
return newObj
}
// Used as fake function for default callback in signals section
func blankNotify() {}
/*************************************************/
/* Images functions, used to initialize objects */
/* You can use it to load your own embedded */
/* images, icons ... */
/*********************************************/
// TryStockIcon: Convenience function to combine a stock-icons-theme
// and an asset/filename icon to send to "GetPixBuf()" function.
func TryStockIcon(iconLookupFlag gtk.IconLookupFlags, stockName, assetName interface{}) (names []interface{}) {
names = append(names, stockName)
names = append(names, iconLookupFlag)
names = append(names, assetName)
return
}
// OptPict: hold options for SetPict()
type OptPict struct {
Size int
// 0 = limit height, 1 = limit width, 2 = percents,
// 3 = limit height or width depending on MaxSize value.
ResizeMethode int
// Max size accepted relative to "ResizeMethode" = 3
MaxSize int
// SpinButton icon position
Position gtk.EntryIconPosition
// Ask for a GdkPixbufAnimation image
Animated bool
// Rotate image / icon
Rotate gdk.PixbufRotation
// Opacity [0.0 to 1.0] work with: Box, ToggleButton, ToolButton, MenuButton, Button
Opacity float64
}
// SetPic: Assign image to an Object depending on type, accept stock-icons-theme,
// filename or []byte. Use OptPict{} struct to set options, size can be applied
// directly as int. The options are described below:
// - Load a gif animated image and specify using animation (true), resizing not
// allowed with animations.
// SetPict(GtkImage, "linearProgressHorzBlue.gif", OptPict{Animated: true})
// - Resize to 32 pixels height, keep porportions & assign image to GtkButton.
// SetPict(GtkButton, "stillImage.png", 32) or
// SetPict(GtkButton, "stillImage.png", OptPict{Size: 32})
// - With default size, resizing not allowed for GtkSpinButton position at right-end.
// SetPict(GtkSpinButton, "stillImage.png", OptPict{Position: gtk.ENTRY_ICON_SECONDARY})
// - Load a stock-icons-theme or an asset/filename icon to a GtkMenuButton where stock not available.
// SetPict(gtk.MenuButton, TryStockIcon("open-menu-symbolic", "stillImage.png"), 18) or
// SetPict(gtk.MenuButton, TryStockIcon("open-menu-symbolic", "stillImage.png"), OptPict{Size: 18})
// - Rotate upsidedown, resize 14px:
// SetPict(gtk.MenuButton, TryStockIcon("open-menu-symbolic", "stillImage.png"), OptPict{Size: 14, Rotate: gdk.PIXBUF_ROTATE_UPSIDEDOWN})
func SetPict(iObject, varPath interface{}, options ...interface{}) (err error) {
var inPixbuf *gdk.Pixbuf
var inPixbufAnimation *gdk.PixbufAnimation
var image *gtk.Image
var opt OptPict
if len(options) > 0 {
switch options[0].(type) {
case int:
opt.Size = options[0].(int)
default:
opt = options[0].(OptPict)
}
}
// Since the default initialization value is set to 0,
// we need to change it to show a visible image ...
if opt.Opacity == 0 {
opt.Opacity = 1
}
// Get pixbuff type (normal or animation for GtkImage)
if opt.Animated {
if inPixbufAnimation, err = GetPixBufAnimation(varPath); err == nil {
image, err = gtk.ImageNewFromAnimation(inPixbufAnimation)
}
} else {
if inPixbuf, err = GetPixBuf(varPath, opt); err == nil {
image, err = gtk.ImageNewFromPixbuf(inPixbuf)
}
}
// Handling the error if there is
if err != nil {
var filename string
// Look for a stock icon or a filename or an ambedded asset
switch path := varPath.(type) {
case []interface{}:
filename = path[2].(string)
default:
filename = path.(string)
}
if _, err = os.Stat(filename); len(filename) > 0 /*os.IsNotExist(err)*/ {
err = fmt.Errorf("SetPict: [%v] %v", varPath, err)
fmt.Println(err.Error()) // double output
}
return
}
// Objects parsing
if image != nil {
image.SetOpacity(opt.Opacity)
switch object := iObject.(type) {
case *gtk.Button: // Set Image to GtkButton
object.SetImage(image)
object.SetAlwaysShowImage(true)
case *gtk.MenuButton: // Set Image to GtkMenuButton
object.SetImage(image)
object.SetAlwaysShowImage(true)
case *gtk.ToggleButton: // Set Image to GtkToggleButton
object.SetImage(image)
object.SetAlwaysShowImage(true)
case *gtk.ToolButton: // Set Image to GtkToolButton
object.SetIconWidget(image)
case *gtk.Box: // Add Image to GtkBox
object.Add(image)
case *gtk.SpinButton: // Set Icon to GtkSpinButton, No resize, No Animate.
object.SetIconFromPixbuf(opt.Position, inPixbuf)
case *gtk.ApplicationWindow: // Set Icon to GtkApplicationWindow, No Animate.
object.SetIcon(inPixbuf)
case *gtk.Window: // Set Icon to GtkWindow, No Animate.
object.SetIcon(inPixbuf)
case *gtk.Image: // Set Image to GtkImage
if opt.Animated {
object.SetFromAnimation(inPixbufAnimation)
} else {
object.SetFromPixbuf(inPixbuf)
}
}
} else {
err = fmt.Errorf("Image error: %v", err)
}
return
}
// GetPixBuf: Get gdk.PixBuf from stock, filename or []byte, depending
// on type. For an explanation of the options, see above, OptPict{}.
// note: If stock-icons-theme does not exist, the asset/filename icon is used.
func GetPixBuf(varPath interface{}, opt OptPict) (outPixbuf *gdk.Pixbuf, err error) {
var pixbufLoader *gdk.PixbufLoader
// Look for a stock icon and a filename or ambedded asset
switch varPath.(type) {
case []interface{}:
var iconTheme *gtk.IconTheme
if iconTheme, err = gtk.IconThemeGetDefault(); err == nil {
stock := varPath.([]interface{})[0].(string)
iconLookupFlags := varPath.([]interface{})[1].(gtk.IconLookupFlags)
outPixbuf, err = iconTheme.LoadIcon(stock, opt.Size, iconLookupFlags)
}
if err != nil {
varPath = varPath.([]interface{})[2]
fmt.Println(err.Error() + ". try to load the secondary choice.")
}
}
if outPixbuf == nil { // in this case, no stock icon was found or was not requested
switch varPath.(type) {
case string: // Its a filename
outPixbuf, err = gdk.PixbufNewFromFile(varPath.(string))
case []uint8: // Its a binary data (embedded asset)
if pixbufLoader, err = gdk.PixbufLoaderNew(); err == nil {
outPixbuf, err = pixbufLoader.WriteAndReturnPixbuf(varPath.([]byte))
}
}
if err == nil && opt.Size != 0 {
newWidth, newHeight := NormalizeSize(outPixbuf.GetWidth(), outPixbuf.GetHeight(), opt)
if outPixbuf, err = outPixbuf.ScaleSimple(newWidth, newHeight, gdk.INTERP_HYPER); err == nil {
outPixbuf, err = outPixbuf.RotateSimple(opt.Rotate)
}
}
}
return
}
// GetPixBufAnimation: Get gdk.PixBufAnimation from filename or []byte, depending on type
func GetPixBufAnimation(varPath interface{}) (outPixbufAnimation *gdk.PixbufAnimation, err error) {
var pixbufLoader *gdk.PixbufLoader
switch varPath.(type) {
case string:
outPixbufAnimation, err = gdk.PixbufAnimationNewFromFile(varPath.(string))
case []uint8:
if pixbufLoader, err = gdk.PixbufLoaderNew(); err == nil {
outPixbufAnimation, err = pixbufLoader.WriteAndReturnPixbufAnimation(varPath.([]byte))
}
}
return
}
// NormalizeSize: compute new size with kept proportions based on defined format.
// formats: 0 = limit height, 1 = limit width, 2 = percents,
// 3 = limit height or width depending on MaxSize value.
func NormalizeSize(oldWidth, oldHeight int, opt OptPict) (outWidth, outHeight int) {
switch opt.ResizeMethode {
case 0: // limit Height
outWidth = int(float64(oldWidth) * (float64(opt.Size) / float64(oldHeight)))
outHeight = opt.Size
case 1: // limit Width
outWidth = opt.Size
outHeight = int(float64(oldHeight) * (float64(opt.Size) / float64(oldWidth)))
case 2: // percent
outWidth = int((float64(oldWidth) * float64(opt.Size)) / 100)
outHeight = int((float64(oldHeight) * float64(opt.Size)) / 100)
case 3: // limit Height or Width
switch {
case oldWidth >= opt.MaxSize:
opt.Size = opt.MaxSize
opt.ResizeMethode = 1
return NormalizeSize(oldWidth, oldHeight, opt)
case oldHeight >= opt.MaxSize:
opt.Size = opt.MaxSize
opt.ResizeMethode = 0
return NormalizeSize(oldWidth, oldHeight, opt)
}
opt.ResizeMethode = 0
return NormalizeSize(oldWidth, oldHeight, opt)
}
return
}
/***************************************/
/* Embedded data conversion functions */
/* Used to make variable content */
/* available in go-source */
/***********************************/
// getBytesFromVarAsset: Get []byte representation from file or asset, depending on type
func getBytesFromVarAsset(varPath interface{}) (outBytes []byte, err error) {
switch reflect.TypeOf(varPath).String() {
case "string":
return ioutil.ReadFile(varPath.(string))
case "[]uint8":
return varPath.([]byte), err
}
return
}
// HexToBytes: Convert Gzip Hex to []byte used for embedded binary in source code
func HexToBytes(varPath string, gzipData []byte) (outByte []byte) {
r, err := gzip.NewReader(bytes.NewBuffer(gzipData))
if err == nil {
var bBuffer bytes.Buffer
if _, err = io.Copy(&bBuffer, r); err == nil {
if err = r.Close(); err == nil {
return bBuffer.Bytes()
}
}
} else {
fmt.Printf("An error occurred while reading: %s\n%s\n", varPath, err.Error())
}
return
}
/*******************************/
/* Simplified files Functions */
/*****************************/
// Make temporary directory
func tempMake(prefix string) (dir string) {
var err error
if dir, err = ioutil.TempDir("", prefix+"-"); err != nil {
log.Fatal(err)
}
return dir + string(os.PathSeparator)
}
// Retrieve current realpath and options filename. Options/Config path
// depend on devMode value, true means actual directory, false means
// $USER/.config/Creat/appName/*
func getAbsRealPath() (absoluteRealPath, optFilename string) {
var err error
var cUser *user.User
var base, absoluteBaseName string
var setExt = func(filename, ext string) (out string) {
return filename[:len(filename)-len(path.Ext(filename))] + ext
}
if absoluteBaseName, err = os.Executable(); err == nil {
absoluteRealPath, base = filepath.Split(absoluteBaseName)
configPath := absoluteRealPath
baseNoExt := setExt(base, "")
if !devMode {
if cUser, err = user.Current(); err == nil {
configPath = filepath.Join(cUser.HomeDir, ".config", ToCamel(Creat), baseNoExt)
if _, err = os.Stat(configPath); os.IsNotExist(err) {
err = os.MkdirAll(configPath, 0755)
}
}
}
if err == nil {
optFilename = setExt(filepath.Join(configPath, baseNoExt), ".opt")
}
}
if err != nil {
log.Fatal(err)
}
return
}
// ToCamel: Turn string into camel case
func ToCamel(inString string, lowerAtFirst ...bool) (outString string) {
var laf bool
if len(lowerAtFirst) != 0 {
laf = lowerAtFirst[0]
}
nonAlNum := regexp.MustCompile("[[:punct:][:space:]]")
tmpString := nonAlNum.Split(inString, -1)
for idx, word := range tmpString {
if laf && idx < 1 {
outString += strings.ToLower(word)
} else {
outString += strings.Title(word)
}
}
return outString
}