-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpopup.js
425 lines (349 loc) · 13 KB
/
popup.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
// i dont give fuck about code quality, its just a quick pet project to improve my QOL
// maybe someday I'll find the time and make it with better code and design, but right now i don't give a shit ¯\_(ツ)_/¯
document.addEventListener("DOMContentLoaded", () => {
const tabsList = document.querySelector(".tabs-list")
let highlightedTab
chrome.tabs.query({ windowId: chrome.windows.WINDOW_ID_CURRENT }, (tabs) => {
tabs.forEach(CreateTab)
highlightedTab.scrollIntoView({ behavior: "instant", block: "center" })
setTimeout(() => {
highlightedTab.scrollIntoView({ behavior: "instant", block: "center" })
}, 100)
})
const colors = {
grey: "#DADCE0",
blue: "#8AB4F8",
red: "#F28B82",
yellow: "#FDD663",
green: "#81C995",
pink: "#FF8BCB",
purple: "#C58AF9",
cyan: "#78D9EC",
orange: "#FCAD70",
}
let groups = {}
let tabs = []
let search = document.querySelector(".search-bar > input")
search.addEventListener("keyup", SearchBar)
// needle characters can be at a distance
function Fuzzy1(needle, haystack, maxDist = 3) {
const needleLen = needle.length
const haystackLen = haystack.length
let i = 0
let j = 0
let dist = 0
let matches = 0
while (i < needleLen && j < haystackLen) {
if (needle[i] === haystack[j]) {
matches++
i++
j++
dist = 0
} else {
j++
dist++
}
if (dist > maxDist) {
i = 0 // Reset needle pointer
matches = 0 // Reset matches counter
j -= dist - 1 // Move haystack pointer back
dist = 0 // Reset distance counter
}
}
return matches === needleLen
}
// needle can contain several words, the search for which will work separately
function Fuzzy2(needle, haystack) {
const needleWords = needle.split(/\s+/).filter((word) => word.length > 0)
const haystackWords = haystack
.split(/\s+/)
.filter((word) => word.length > 0)
for (let i = 0; i < needleWords.length; i++) {
const needleWord = needleWords[i].toLowerCase()
let found = false
for (let j = 0; j < haystackWords.length; j++) {
const haystackWord = haystackWords[j].toLowerCase()
if (Fuzzy1(needleWord, haystackWord)) {
found = true
break
}
}
if (!found) {
return false
}
}
return true
}
function FzySearch(userinput, tab) {
let title = tab.tab.title.toLowerCase()
let url = tab.tab.url.toLowerCase()
return (
title.includes(userinput) ||
url.includes(userinput) ||
// Fuzzy1(userinput, title) ||
// Fuzzy1(userinput, url) ||
Fuzzy2(userinput, title) ||
Fuzzy2(userinput, url)
)
}
function SearchBar(event) {
// let userinput = event.target.value.trim().toLowerCase()
let userinput = event.target.value.trim().toLowerCase()
let visibleGroups = {}
tabs.forEach((tab) => {
const visible = userinput.length == 0 || FzySearch(userinput, tab)
if (visible) {
tab.node.style.display = ""
visibleGroups[tab.tab.groupId] = true
} else {
tab.node.style.display = "none"
}
})
Object.keys(groups).forEach((key) => {
const group = groups[key]
if (userinput.length == 0) {
if (group.data.collapsed) {
group.li.classList.add("collapsed")
} else {
group.li.classList.remove("collapsed")
}
} else if (visibleGroups[group.id]) {
group.li.classList.remove("collapsed")
} else {
group.li.classList.add("collapsed")
}
})
}
function CreateGroup(groupId) {
if (groups[groupId] !== undefined) {
return groups[groupId].ul
}
const li = document.createElement("li")
tabsList.appendChild(li)
li.classList.add("tabs-element")
li.onclick = (e) => {
e.preventDefault()
li.classList.toggle("collapsed")
}
const span = document.createElement("span")
li.appendChild(span)
span.textContent = groupId
span.style.userSelect = "none"
const ul = document.createElement("ul")
li.appendChild(ul)
ul.classList.add("tabs-group")
let obj = {
id: groupId,
ul: ul,
li: li,
data: {},
}
chrome.tabGroups.get(groupId, (data) => {
span.style.backgroundColor = colors[data.color] || data.color
span.textContent = data.title
if (data.collapsed) {
li.classList.add("collapsed")
}
obj.data = data
})
groups[groupId] = obj
return ul
}
function SetNewtabIcon(img) {
img.src =
""
}
function AppendTab(tab, parent) {
const li = document.createElement("li")
parent.appendChild(li)
li.classList.add("tab")
li.style.cursor = "pointer"
li.onclick = () => {
chrome.tabs.update(tab.id, {
active: true,
})
}
tabs.push({
tab: tab,
node: li,
parent: parent,
})
const title = document.createElement("div")
title.classList.add("tab-title")
li.appendChild(title)
if (tab.pinned) {
const img = document.createElement("img")
img.alt = "Pinned"
img.classList.add("small")
title.appendChild(img)
img.src =
"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='24' height='24' viewBox='0 0 24 24'%3E%3Cg fill='%235F6368'%3E%3Cpath fill='%235F6368' d='M6.672,15.914l-5.379,5.379c-0.391,0.391-0.391,1.023,0,1.414C1.488,22.902,1.744,23,2,23 s0.512-0.098,0.707-0.293l5.379-5.379L6.672,15.914z'%3E%3C/path%3E %3Cpath fill='%235F6368' d='M23.707,7.293l-7-7c-0.391-0.391-1.023-0.391-1.414,0s-0.391,1.023,0,1.414l0.908,0.908L5.18,8.766 L3.707,7.293c-0.391-0.391-1.023-0.391-1.414,0s-0.391,1.023,0,1.414l13,13C15.488,21.902,15.744,22,16,22s0.512-0.098,0.707-0.293 c0.391-0.391,0.391-1.023,0-1.414l-1.473-1.473l6.151-11.021l0.908,0.908C22.488,8.902,22.744,9,23,9s0.512-0.098,0.707-0.293 C24.098,8.316,24.098,7.684,23.707,7.293z'%3E%3C/path%3E%3C/g%3E%3C/svg%3E"
}
let speakerIcons = {
[true]:
"data:image/svg+xml,%3Csvg width='514' height='514' viewBox='0 0 514 514' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath fill-rule='evenodd' clip-rule='evenodd' d='M91.0847 152.096H47.0223C21.6049 152.096 1 172.683 1 198.079L1.00023 317.348C1.00023 342.742 21.6051 363.33 47.0225 363.33H91.8942L216.437 475.166C224.611 482.5 240.684 492.271 257.933 483.588C274.181 475.412 277.134 457.681 277.134 446.818V338.145L91.0847 152.096ZM388.709 333.755L347.913 292.959C356.44 285.973 363.426 273.51 363.426 257C363.426 234.285 350.199 219.228 337.929 215.561L332.416 213.915C320.241 210.278 313.321 197.467 316.962 185.302L320.257 174.288C323.897 162.122 336.719 155.209 348.894 158.846L354.408 160.492C395.068 172.638 420.953 213.349 420.953 257C420.953 286.602 409.05 314.849 388.709 333.755ZM341.903 402.915C339.124 407.991 338.248 414.114 339.928 420.127L343.019 431.199C346.436 443.431 359.129 450.579 371.368 447.167L376.91 445.622C378.901 445.066 380.879 444.477 382.841 443.853L341.903 402.915ZM454.813 399.86L414.127 359.174C439.235 334.3 455.472 298.142 455.472 257.003C455.472 191.387 414.167 138.443 361.448 123.745L355.907 122.2C343.665 118.787 336.511 106.106 339.928 93.8757L343.019 82.8032C346.436 70.5726 359.129 63.4247 371.368 66.8375L376.91 68.3822C456.521 90.5781 513 167.756 513 257.003C513 312.853 490.88 363.979 454.813 399.86ZM277.134 222.18L152.189 97.2349L216.437 38.8516C224.583 31.5395 240.663 21.7179 257.935 30.406C274.207 38.5901 277.134 56.3607 277.134 67.2001V222.18Z' fill='%23B4B8BD'/%3E%3Crect x='-8.20508' y='32.8066' width='58' height='691.981' rx='22' transform='rotate(-45 -8.20508 32.8066)' fill='%23B4B8BD'/%3E%3C/svg%3E%0A",
[false]:
"data:image/svg+xml,%3Csvg width='513' height='513' viewBox='0 0 513 513' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M276.134 66.2001C276.134 55.3607 273.207 37.5901 256.935 29.406C239.663 20.7179 223.583 30.5395 215.437 37.8516L90.816 151.096H46.0223C20.6049 151.096 0 171.683 0 197.079L0.000230424 316.348C0.000230424 341.742 20.6051 362.33 46.0225 362.33H90.8942L215.437 474.166C223.611 481.5 239.684 491.271 256.933 482.588C273.181 474.412 276.134 456.681 276.134 445.818V66.2001Z' fill='%23B4B8BD'/%3E%3Cpath d='M375.91 67.3822L370.368 65.8375C358.129 62.4247 345.436 69.5726 342.019 81.8032L338.928 92.8757C335.511 105.106 342.665 117.788 354.907 121.2L360.448 122.745C413.167 137.443 454.472 190.387 454.472 256.003C454.472 321.617 413.167 374.561 360.448 389.259L354.907 390.802C342.665 394.216 335.511 406.898 338.928 419.127L342.019 430.199C345.436 442.431 358.129 449.579 370.368 446.167L375.91 444.622C455.521 422.426 512 345.247 512 256.003C512 166.756 455.521 89.5781 375.91 67.3822Z' fill='%23B4B8BD'/%3E%3Cpath d='M353.408 159.492L347.894 157.846C335.719 154.209 322.897 161.122 319.257 173.288L315.962 184.302C312.321 196.467 319.241 209.278 331.416 212.915L336.929 214.561C349.199 218.228 362.426 233.285 362.426 256C362.426 278.717 349.199 293.774 336.929 297.439L331.416 299.085C319.241 302.722 312.321 315.533 315.962 327.698L319.257 338.713C322.897 350.877 335.719 357.791 347.894 354.154L353.408 352.507C394.068 340.361 419.953 299.653 419.953 256C419.953 212.349 394.068 171.638 353.408 159.492Z' fill='%23B4B8BD'/%3E%3C/svg%3E%0A",
}
let audible
if (tab.audible) {
const img = document.createElement("img")
img.alt = "Audible"
img.classList.add("small")
title.appendChild(img)
img.src = speakerIcons[tab.mutedInfo.muted]
audible = img
}
const img = document.createElement("img")
img.alt = "Favicon"
title.appendChild(img)
if (tab.favIconUrl) {
// img.src = "chrome://favicon/" + tab.url
// img.onerror = () => {
img.src = tab.favIconUrl
img.onerror = () => {
img.src = "https://images.duckduckgo.com/iu/?u=" + tab.favIconUrl
img.onerror = () => {
SetNewtabIcon(img)
console.log(tab.url, img, img.src)
}
}
// }
} else {
SetNewtabIcon(img)
}
if (tab.highlighted) {
li.classList.add("highlighted")
highlightedTab = li
}
const span = document.createElement("span")
title.appendChild(span)
span.textContent = tab.title
const desc = document.createElement("div")
desc.classList.add("tab-desc")
desc.textContent = tab.url
li.appendChild(desc)
li.addEventListener("contextmenu", (e) => {
e.preventDefault()
chrome.bookmarks.search({ url: tab.url }, (results) => {
let favorites
if (results.length > 0) {
favorites = {
name: "Remove from favorites",
cback: () => {
chrome.bookmarks.remove(results[0].id)
},
}
} else {
favorites = {
name: "Add to favorites",
cback: () => {
chrome.bookmarks.create({
title: tab.title,
url: tab.url,
})
},
}
}
const menu = ContextMenu([
{
name: "Copy URL",
cback: () => {
navigator.clipboard.writeText(tab.url)
},
},
{
name: tab.mutedInfo.muted ? "Unmute" : "Mute",
cback: () => {
tab.mutedInfo.muted = !tab.mutedInfo.muted
chrome.tabs.update(tab.id, {
muted: tab.mutedInfo.muted,
})
audible.src = speakerIcons[tab.mutedInfo.muted]
},
customcheck: () => {
return tab.audible
},
},
favorites,
{
name: "Close",
cback: () => {
chrome.tabs.remove(tab.id)
li.remove()
},
warning: true,
},
])
SetContextMenuPosition(menu, e)
})
})
}
function CreateTab(tab) {
if (tab.groupId === -1) {
AppendTab(tab, tabsList)
} else {
AppendTab(tab, CreateGroup(tab.groupId))
}
}
let w8ContextMenu = false
function ContextMenu(items) {
const prev = document.querySelector("#context-menu")
if (prev !== null) {
prev.remove()
}
const menu = document.createElement("ul")
menu.id = "context-menu"
items.forEach((item) => {
if (item.customcheck && item.customcheck() == false) {
return
}
const li = document.createElement("li")
li.innerHTML = item.name
li.addEventListener("click", () => {
item.cback()
menu.remove()
})
if (item.warning) {
li.classList.add("warning")
}
menu.appendChild(li)
})
document.body.appendChild(menu)
w8ContextMenu = true
setTimeout(() => {
w8ContextMenu = false
}, 10)
return menu
}
function SetContextMenuPosition(menu, e) {
let w = menu.clientWidth
let h = menu.clientHeight
let x = e.pageX
let y = e.pageY
// let x = Math.min(e.pageX, document.body.scrollWidth - w)
// let y = Math.min(e.pageY, document.body.scrollHeight - h)
if (x + w > document.body.scrollWidth) {
x = x - w
}
if (y + h > document.body.scrollHeight) {
y = y - h
}
menu.style.left = `${x}px`
menu.style.top = `${y}px`
}
document.addEventListener("click", ContextMenuRemover)
document.addEventListener("contextmenu", ContextMenuRemover)
function ContextMenuRemover(e) {
const menu = document.querySelector("#context-menu")
if (
w8ContextMenu == true ||
menu == null ||
e.target == menu ||
menu.contains(e.target)
) {
return
}
menu.remove()
}
})