-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathno-background.js
542 lines (469 loc) · 14.7 KB
/
no-background.js
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
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
// Variables used by Scriptable.
// These must be at the very top of the file. Do not edit.
// icon-color: purple; icon-glyph: chess-board;
/* -----------------------------------------------
Script : no-background.js
Author : dev@supermamon.com
Version : 1.8.0
Description :
A module to create illusions of transparent
backgrounds for Scriptable widgets
Adapted from Max Zeryck's (@mzeryck) amazing
invisible widget shared on the Automtors discourse
https://talk.automators.fm/t/widget-examples/7994/135
Changelog :
v1.8.0
- (new) BREAKING CHANGE: background assignments
are unique per device so that each device can
have it's own wallpaper. Run "Regenerate Slices"
on each device.
But, there's a bug the current Scriptable version
where Device.name() returns the model instead.
So, the wallppapers actually can't be unique
yet. Affects stable version 1.6.12 or beta 1.7
- (new) support for iPhone 14 Pro/Pro Max.
Thank you @mzeryck & u/Senion-Ad-883
v1.7.0
- (new) support for iPhone 12 Mini. Thanks @blacksector
v1.6.0
- (new) `transparent` and `for` alias to `getSliceForWidget`
- (new) merged `No Background Config` code to maintain 1 code
- (new) auto-detect iCloud usage
v1.5.0
- (update) iPhone 12 Pro Max compatibility
v1.4.0
- (update) also prompt for setup on the getPathForSlice method
v1.3.0
- (update) automativally prompt for setup when
slices are missing or if the widget's
position is not yet stored.
v1.2.0
- (new) applyTint method to simulate a
semi-tranparent look
v1.1.1
- (fix) syntax error on generateSlices
- (fix) incorrect iPhone 12 size
v1.1.0
- Support for iPhone 12 & 12 Pro
v1.0.2
- Typos on the documentation
v1.0.1
- Fix iamge does not exists issue
----------------------------------------------- */
const ALERTS_AS_SHEETS = false
const USES_ICLOUD = usesiCloud()
const fm = FileManagerAdaptive()
const CACHE_FOLDER = 'cache/nobg'
const cachePath = fm.joinPath(fm.documentsDirectory(), CACHE_FOLDER);
const device_id = `${Device.model()}_${Device.name()}`.replace(/[^a-zA-Z0-9\-\_]/, '').toLowerCase()
const _exports_ = {}
//------------------------------------------------
_exports_.cachePath = cachePath
//------------------------------------------------
_exports_.generateSlices = async function ({ caller = 'none' }) {
const opts = { caller }
let appearance = (await isUsingDarkAppearance()) ? 'dark' : 'light'
let altAppearance = appearance == 'dark' ? 'light' : 'dark'
if (!fm.fileExists(cachePath)) {
fm.createDirectory(cachePath, true)
}
let message;
message = "To change background make sure you have a screenshot of you home screen. Go to your home screen and enter wiggle mode. Scroll to the empty page on the far right and take a screenshot."
let options = ["Pick Screenshot", "Exit to Take Screenshot"]
let resp = await presentAlert(message, options, ALERTS_AS_SHEETS)
if (resp == 1) return false
// Get screenshot and determine phone size.
let wallpaper = await Photos.fromLibrary()
let height = wallpaper.size.height
let suffix = "";
// Check for iPhone 12 Mini here:
if (height == 2436) {
// We'll save everything in the config, to keep things centralized
let cfg = await loadConfig()
if (cfg["phone-model"] === undefined) {
// Doesn't exist, ask them which phone they want to generate for,
// the mini or the others?
message = "What model of iPhone do you have?"
let options = ["iPhone 12 mini", "iPhone 11 Pro, XS, or X"]
resp = await presentAlert(message, options, ALERTS_AS_SHEETS)
// 0 represents iPhone Mini and 1 others.
cfg["phone-model"] = resp
await saveConfig(cfg) // Save the config
if (resp === 0) {
suffix = "_mini";
}
} else {
// Config already contains iPhone model, use it from cfg
if (cfg["phone-model"]) {
suffix = "_mini";
}
}
}
let phone = phoneSizes[height + suffix]
if (!phone) {
message = "It looks like you selected an image that isn't an iPhone screenshot, or your iPhone is not supported. Try again with a different image."
await presentAlert(message, ["OK"], ALERTS_AS_SHEETS)
return false
}
const families = ['small', 'medium', 'large']
// generate crop rects for all sizes
for (var i = 0; i < families.length; i++) {
const widgetSize = families[i]
let crops = widgetPositions[widgetSize].map(posName => {
let position = posName.toLowerCase().replace(' ', '-')
let crop = { pos: position, w: "", h: "", x: "", y: "" }
crop.w = phone[widgetSize].w
crop.h = phone[widgetSize].h
crop.x = phone.left
let pos = position.split('-')
crop.y = phone[pos[0]]
if (widgetSize == 'large' && pos[0] == 'bottom') {
crop.y = phone['middle']
}
if (pos.length > 1) {
crop.x = phone[pos[1]]
}
return crop
})
for (var c = 0; c < crops.length; c++) {
const crop = crops[c]
const imgCrop = cropImage(wallpaper, new Rect(crop.x, crop.y, crop.w, crop.h))
const imgName = `${device_id}-${appearance}-${widgetSize}-${crop.pos}.jpg`
const imgPath = fm.joinPath(cachePath, imgName)
if (fm.fileExists(imgPath)) {
// sometimes it wouldn't overwrite.
// so better delete the file first
if (USES_ICLOUD) await fm.downloadFileFromiCloud(imgPath)
try { fm.remove(imgPath) } catch (e) { }
}
fm.writeImage(imgPath, imgCrop)
}
}
if (opts.caller != 'self') {
message = `Slices saved for ${appearance} mode. You can switch to ${altAppearance} mode and run this again to also generate slices.`
} else {
message = 'Slices saved.'
}
await presentAlert(message, ["Ok"], ALERTS_AS_SHEETS)
return true
}
//------------------------------------------------
_exports_.applyTint = function (widget, tint, alpha) {
tint = tint || '#ffffff'
alpha = alpha || 0.2
const col = new Color(tint, alpha)
let gradient = new LinearGradient()
gradient.locations = [0, 1]
gradient.colors = [col, col]
widget.backgroundGradient = gradient
}
//------------------------------------------------
_exports_.getSlice = async function (name) {
let appearance = (await isUsingDarkAppearance())
? 'dark'
: 'light'
let position = name
//log(position)
const imgPath = fm.joinPath(cachePath, `${device_id}-${appearance}-${position}.jpg`)
if (!fm.fileExists(imgPath)) {
log('image does not exists. setup required.')
var setupCompleted = await _exports_.generateSlices({ caller: 'getSliceForWidget' })
if (!setupCompleted) {
return null
}
}
if (USES_ICLOUD) await fm.downloadFileFromiCloud(imgPath)
let image = fm.readImage(imgPath)
return image
}
//------------------------------------------------
_exports_.getPathForSlice = async function (slice_name) {
let appearance = (await isUsingDarkAppearance())
? 'dark'
: 'light'
let imgPath = fm.joinPath(cachePath,
`${device_id}-${appearance}-${slice_name}.jpg`)
let fileExists = fm.fileExists(imgPath)
if (!fileExists) {
fileExists = await _exports_.generateSlices('self')
} else {
imgPath = null
}
if (USES_ICLOUD && fileExists) await fm.downloadFileFromiCloud(imgPath)
return imgPath
}
//------------------------------------------------
_exports_.getSliceForWidget = async function (
instance_name,
reset = false) {
let appearance = (await isUsingDarkAppearance())
? 'dark'
: 'light'
var cfg = await loadConfig()
var position = reset ? null : cfg[instance_name]
if (!position) {
log(`Background for "${instance_name}" is not yet set.`)
// check if slices exists
const testImage = fm.joinPath(cachePath, `${device_id}-${appearance}-medium-top.jpg`)
let readyToChoose = false
if (!fm.fileExists(testImage)) {
// need to generate slices
readyToChoose = await _exports_.generateSlices({ caller: 'self' })
} else {
readyToChoose = true
}
// now set the
if (readyToChoose) {
var backgrounChosen = await _exports_.chooseBackgroundSlice(instance_name)
}
if (backgrounChosen) {
cfg = await loadConfig()
position = cfg[instance_name]
} else {
return null
}
}
const imgPath = fm.joinPath(cachePath, `${device_id}-${appearance}-${position}.jpg`)
if (!fm.fileExists(imgPath)) {
log(`Slice does not exists - ${device_id}-${appearance}-${position}.jpg`)
return null
}
if (USES_ICLOUD) await fm.downloadFileFromiCloud(imgPath)
let image = fm.readImage(imgPath)
return image
}
//------------------------------------------------
_exports_.transparent = _exports_.getSliceForWidget
//------------------------------------------------
_exports_.chooseBackgroundSlice = async function (name) {
// Prompt for widget size and position.
let message = "What is the size of the widget?"
let sizes = ["Small", "Medium", "Large", "Cancel"]
let size = await presentAlert(message, sizes, ALERTS_AS_SHEETS)
if (size == 3) return false
let widgetSize = sizes[size].toLowerCase()
message = "Where will it be placed on?"
let positions = widgetPositions[widgetSize]
positions.push('Cancel')
let resp = await presentAlert(message, positions, ALERTS_AS_SHEETS)
if (resp == positions.length - 1) return false
let position = positions[resp].toLowerCase().replace(' ', '-')
let cfg = await loadConfig()
cfg[name] = `${widgetSize}-${position}`
await saveConfig(cfg)
await presentAlert("Background saved.", ["Ok"], ALERTS_AS_SHEETS)
return true
}
//------------------------------------------------
_exports_.resetConfig = async function () {
await saveConfig({})
log('config file cleared')
}
_exports_.loadConfig = loadConfig
_exports_.clean = () => {
if (fm.fileExists(cachePath)) {
fm.remove(cachePath)
}
}
//-- [helpers] -----------------------------------
//------------------------------------------------
async function loadConfig() {
const configPath = fm.joinPath(cachePath, "widget-positions.json")
if (!fm.fileExists(cachePath)) {
fm.createDirectory(cachePath)
}
if (!fm.fileExists(configPath)) {
await fm.writeString(configPath, "{}")
return {}
} else {
if (USES_ICLOUD) await fm.downloadFileFromiCloud(configPath)
const strConf = fm.readString(configPath)
const cfg = JSON.parse(strConf)
return cfg
}
}
//------------------------------------------------
async function saveConfig(cfg) {
const configPath = fm.joinPath(
cachePath,
"widget-positions.json")
if (USES_ICLOUD) {
await fm.downloadFileFromiCloud(configPath)
}
await fm.writeString(configPath,
JSON.stringify(cfg))
return cfg
}
//------------------------------------------------
async function presentAlert(prompt = ""
, items = ["OK"]
, asSheet = false) {
let alert = new Alert()
alert.message = prompt
for (var n = 0; n < items.length; n++) {
alert.addAction(items[n])
}
let resp = asSheet
? await alert.presentSheet()
: await alert.presentAlert()
return resp
}
//------------------------------------------------
const widgetPositions = {
"small": [
"Top Left", "Top Right",
"Middle Left", "Middle Right",
"Bottom Left", "Bottom Right"
],
"medium": ["Top", "Middle", "Bottom"],
"large": ["Top", "Bottom"]
}
//------------------------------------------------
const phoneSizes = {
"2796": {
"models": ["14 Pro Max"],
"small": { "w": 510, "h": 510 },
"medium": { "w": 1092, "h": 510 },
"large": { "w": 1092, "h": 1146 },
"left": 99,
"right": 681,
"top": 282,
"middle": 918,
"bottom": 1554
},
"2556": {
"models": ["14 Pro"],
"small": { "w": 474, "h": 474 },
"medium": { "w": 1014, "h": 474 },
"large": { "w": 1014, "h": 1062 },
"left": 82,
"right": 622,
"top": 270,
"middle": 858,
"bottom": 1446
},
"2778": {
"models": ["12 Pro Max", "13 Pro Max", "14 Plus"],
"small": { "w": 510, "h": 510 },
"medium": { "w": 1092, "h": 510 },
"large": { "w": 1092, "h": 1146 },
"left": 96,
"right": 678,
"top": 246,
"middle": 882,
"bottom": 1518
},
"2532": {
"models": ["12", "12 Pro", "13", "14"],
"small": { "w": 474, "h": 474 },
"medium": { "w": 1014, "h": 474 },
"large": { "w": 1014, "h": 1062 },
"left": 78,
"right": 618,
"top": 231,
"middle": 819,
"bottom": 1407
},
"2688": {
"models": ["Xs Max", "11 Pro Max"],
"small": { "w": 507, "h": 507 },
"medium": { "w": 1080, "h": 507 },
"large": { "w": 1080, "h": 1137 },
"left": 81,
"right": 654,
"top": 228,
"middle": 858,
"bottom": 1488
},
"1792": {
"models": ["11", "Xr"],
"small": { "w": 338, "h": 338 },
"medium": { "w": 720, "h": 338 },
"large": { "w": 720, "h": 758 },
"left": 54,
"right": 436,
"top": 160,
"middle": 580,
"bottom": 1000
},
"2436": {
"models": ["X", "Xs", "11 Pro"],
"small": { "w": 465, "h": 465 },
"medium": { "w": 987, "h": 465 },
"large": { "w": 987, "h": 1035 },
"left": 69,
"right": 591,
"top": 213,
"middle": 783,
"bottom": 1353
},
"2436_mini": {
"models": ["12 Mini"],
"small": { "w": 465, "h": 465 },
"medium": { "w": 987, "h": 465 },
"large": { "w": 987, "h": 1035 },
"left": 69,
"right": 591,
"top": 231,
"middle": 801,
"bottom": 1371
},
"2208": {
"models": ["6+", "6s+", "7+", "8+"],
"small": { "w": 471, "h": 471 },
"medium": { "w": 1044, "h": 471 },
"large": { "w": 1044, "h": 1071 },
"left": 99,
"right": 672,
"top": 114,
"middle": 696,
"bottom": 1278
},
"1334": {
"models": ["6", "6s", "7", "8"],
"small": { "w": 296, "h": 296 },
"medium": { "w": 642, "h": 296 },
"large": { "w": 642, "h": 648 },
"left": 54,
"right": 400,
"top": 60,
"middle": 412,
"bottom": 764
},
"1136": {
"models": ["5", "5s", "5c", "SE"],
"small": { "w": 282, "h": 282 },
"medium": { "w": 584, "h": 282 },
"large": { "w": 584, "h": 622 },
"left": 30,
"right": 332,
"top": 59,
"middle": 399,
"bottom": 399
}
}
//------------------------------------------------
function cropImage(img, rect) {
let draw = new DrawContext()
draw.size = new Size(rect.width, rect.height)
draw.drawImageAtPoint(img, new Point(-rect.x, -rect.y))
return draw.getImage()
}
//------------------------------------------------
async function isUsingDarkAppearance() {
return !(Color.dynamic(Color.white(), Color.black()).red)
}
//------------------------------------------------
function usesiCloud() {
return module.filename
.includes('Documents/iCloud~')
}
//------------------------------------------------
function FileManagerAdaptive() {
return module.filename
.includes('Documents/iCloud~')
? FileManager.iCloud()
: FileManager.local()
}
module.exports = _exports_
// -- END OF MODULE CODE --