-
Notifications
You must be signed in to change notification settings - Fork 69
/
Copy pathscript.js
572 lines (518 loc) · 25.8 KB
/
script.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
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
const DEBUG = false;
const config = document.getElementById("eight-dollars-settings");
const data = JSON.parse(config.innerText);
const verifiedUsers = document.getElementById("eight-dollars-verified-users");
const verifiedUsersData = JSON.parse(verifiedUsers.innerText);
const parsedVerifiedUsersData = verifiedUsersData.map((user) =>
user.toLowerCase().replace("\r", "")
);
const VERIFIED_USERS = new Set(parsedVerifiedUsersData);
console.log(VERIFIED_USERS);
const MEME_MODE = data.memeMode;
const TEXT_ENABLED = data.textEnabled;
const REMOVE_TWITTER_BLUE_VERIFICATION = data.removeBlueVerification;
const TEXT_VERIFIED_LABEL = data.textOptions?.verifiedLabel || "";
const TEXT_TWITTER_BLUE_LABEL = data.textOptions?.twitterBlueLabel || "";
const TEXT_ENABLE_BORDER = data.textOptions?.enableBorder ?? true;
const VERIFIED_ACCOUNT_ARIA_LABEL_I18N = {
"ar-x-fm": "حساب موَثَّق",
ar: "حساب موَثَّق",
bg: "Потвърден профил",
bn: "যাচাইকৃত অ্যাকাউন্ট",
ca: "Compte verificat",
cs: "Ověřený účet",
da: "Verificeret konto",
de: "Verifizierter Account",
el: "Επαληθευμένος λογαριασμός",
"en-gb": "Verified account",
en: "Verified account",
es: "Cuenta verificada",
eu: "Egiaztatutako kontua",
fa: "حساب تاییدشده",
fi: "Varmennettu tili",
fil: "Beripikadong account",
fr: "Compte certifié",
ga: "Cuntas deimhnithe",
gl: "Conta verificada",
gu: "ચકાસાયેલું એકાઉન્ટ",
he: "חשבון מאומת",
hi: "सत्यापित खाता",
hr: "Potvrđeni račun",
hu: "Hitelesített felhasználó",
id: "Akun terverifikasi",
it: "Account verificato",
ja: "認証済みアカウント",
kn: "ಪರಿಶೀಲಿಸಿದ ಖಾತೆ",
ko: "인증된 계정",
mr: "खाते सत्यापित केले",
ms: "Akaun Disahkan",
nb: "Verifisert konto",
nl: "Geverifieerd account",
pl: "Zweryfikowane konto",
pt: "Conta verificada",
ro: "Cont verificat",
ru: "Подлинная учетная запись",
sk: "Overený účet",
sr: "Потврђен налог",
sv: "Verifierat konto",
ta: "சரிபார்க்கப்பட்ட கணக்கு",
th: "บัญชีที่ยืนยันแล้ว",
tr: "Onaylanmış hesap",
uk: "Підтверджений профіль",
ur: "تصدیق شدہ اکاؤنٹ",
vi: "Tài khoản đã xác nhận",
"zh-hant": "已認證的帳戶",
zh: "认证账号",
};
const PROVIDES_DETAILS_ARIA_LABEL_I18N = {
"ar-x-fm": "لتوفير تفاصيل حول الحسابات الموثّقة.",
ar: "لتوفير تفاصيل حول الحسابات الموثّقة.",
bg: "Дава подробности за потвърдените профили.",
bn: "যাচাই করা অ্যাকাউন্ট সম্পর্কে বিশদ বিবরণ দিন।",
ca: "Proporciona informació sobre els comptes verificats.",
cs: "Poskytuje podrobnosti o ověřených účtech.",
da: "Giver oplysninger om verificerede konti.",
de: "Gibt Details über verifizierte Accounts an.",
el: "Παρέχει λεπτομέρειες σχετικά με τους επαληθευμένους λογαριασμούς.",
"en-gb": "Provides details about verified accounts.",
en: "Provides details about verified accounts.",
es: "Proporciona detalles sobre las cuentas verificadas.",
eu: "Egiaztatutako kontuei buruzko xehetasunak ematen ditu.",
fa: "جزئیاتی درباره حسابهای کاربری تأییدشده ارائه میدهد.",
fi: "Tarkempia tietoja varmennetuista tileistä.",
fil: "Nagbibigay ng mga detalye tungkol sa mga beripikadong account.",
fr: "Donne des détails sur les comptes certifiés.",
ga: "Soláthraítear sonraí maidir le cuntais dheimhnithe.",
gl: "Fornece detalles sobre contas verificadas.",
gu: "ચકાસાયેલ એકાઉન્ટ વિષે વિગતો આપે છે.",
he: "מידע לגבי חשבונות מאומתים.",
hi: "सत्यापित खातों के विवरण प्रदान करें.",
hr: "Daje podatke o potvrđenim računima.",
hu: "További részletek az ellenőrzött felhasználói fiókokkal kapcsolatban.",
id: "Memberikan detail tentang akun terverifikasi.",
it: "Fornisce dettagli sugli account verificati.",
ja: "認証済みアカウントについての詳細が表示されます。",
kn: "ಪರಿಶೀಲಿಸಿದ ಖಾತೆಗಳ ಬಗ್ಗೆ ವಿವರಗಳನ್ನು ಒದಗಿಸುತ್ತದೆ.",
ko: "인증된 계정에 대한 세부 정보를 제공합니다.",
mr: "सत्यापित खात्यांविषयी तपशील दिला जातो.",
ms: "Menyediakan butiran tentang akaun disahkan.",
nb: "Gir detaljert informasjon om verifiserte kontoer.",
nl: "Verstrekt informatie over geverifieerde accounts.",
pl: "Więcej informacji na temat weryfikacji kont.",
pt: "Dá detalhes sobre contas verificadas.",
ro: "Oferă detalii despre conturile verificate.",
ru: "Объясняет, что такое подлинные учетные записи.",
sk: "Poskytne podrobnosti o overených účtoch.",
sr: "Наводи детаље о потврђеним налозима.",
sv: "Ger information om verifierade konton.",
ta: "சரிபார்க்கப்பட்ட கணக்குகள் பற்றிய விவரங்களை வழங்குகிறது.",
th: "ระบุรายละเอียดเกี่ยวกับบัญชีที่ยืนยันแล้ว",
tr: "Onaylanmış hesaplar hakkında ayrıntılı bilgi verir.",
uk: "Надає відомості про підтверджені профілі.",
ur: "تصدیق شدہ اکاؤنٹس کے متعلق تفصیلات فراہم کریں۔",
vi: "Cung cấp thông tin chi tiết về tài khoản đã xác minh.",
"zh-hant": "提供已認證帳戶的詳細資料。",
zh: "提供已验证账号的详细信息。",
};
const lang = document.documentElement.lang.toLowerCase();
const VERIFIED_ACCOUNT_ARIA_LABEL =
VERIFIED_ACCOUNT_ARIA_LABEL_I18N[lang] || VERIFIED_ACCOUNT_ARIA_LABEL_I18N.en;
const PROVIDES_DETAILS_ARIA_LABEL =
PROVIDES_DETAILS_ARIA_LABEL_I18N[lang] || PROVIDES_DETAILS_ARIA_LABEL_I18N.en;
const COMIC_SANS_BLUE_DOLLAR_SVG = (
isAriaLabel,
className
) => `<svg class="${className}" ${
isAriaLabel ? `aria-label="${VERIFIED_ACCOUNT_ARIA_LABEL}"` : ""
} ${
isAriaLabel ? `data-eight-dollars-original-classes="${className}"` : ""
} style="margin-left: 2px; max-width: 20px; max-height: 20px;" width="24" height="24" viewBox="0 0 24 24" data-eight-dollars-status="blueVerified" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M22.25 12C22.25 10.57 21.37 9.33 20.06 8.66C20.52 7.27 20.26 5.76 19.25 4.75C18.24 3.74 16.73 3.48 15.34 3.94C14.68 2.63 13.43 1.75 12 1.75C10.57 1.75 9.33 2.63 8.67 3.94C7.27 3.48 5.76 3.74 4.75 4.75C3.74 5.76 3.49 7.27 3.95 8.66C2.64 9.33 1.75 10.57 1.75 12C1.75 13.43 2.64 14.67 3.95 15.34C3.49 16.73 3.74 18.24 4.75 19.25C5.76 20.26 7.27 20.51 8.66 20.06C9.33 21.37 10.57 22.25 12 22.25C13.43 22.25 14.68 21.37 15.34 20.06C16.73 20.51 18.24 20.26 19.25 19.25C20.26 18.24 20.52 16.73 20.06 15.34C21.37 14.67 22.25 13.43 22.25 12Z" fill="#1D9BF0"/>
<path d="M11.3339 10.3033V7.58999C10.1557 7.96788 9.56661 8.55235 9.56661 9.34341C9.56661 9.83215 10.1557 10.1521 11.3339 10.3033ZM12.9147 11.898V15.8281C13.5423 15.6467 14.0663 15.3796 14.4867 15.0269C14.9544 14.6238 15.1883 14.188 15.1883 13.7194C15.1883 12.8427 14.4304 12.2356 12.9147 11.898ZM12.9147 4.64242V5.9197H12.9503C13.5186 5.9197 14.167 6.00283 14.8952 6.16911C15.8484 6.38577 16.325 6.66289 16.325 7.00047C16.325 7.44387 16.0616 7.66556 15.5346 7.66556C15.2978 7.66556 14.8804 7.61266 14.2824 7.50685C13.6903 7.396 13.2345 7.3431 12.9147 7.34813V10.4242C14.2232 10.525 15.2475 10.8903 15.9876 11.5201C16.6625 12.1046 17 12.81 17 13.6363C17 14.649 16.5885 15.488 15.7655 16.153C15.0373 16.7375 14.087 17.123 12.9147 17.3094V19.3198C12.9147 19.5163 12.8378 19.6775 12.6838 19.8035C12.5358 19.9345 12.3464 20 12.1155 20C11.5944 20 11.3339 19.7783 11.3339 19.3349V17.4077C8.44464 17.3724 7 16.735 7 15.4955C7 15.042 7.24571 14.8153 7.73712 14.8153C8.01539 14.8153 8.28182 14.9085 8.53641 15.0949C8.91533 15.3721 9.20249 15.5535 9.39787 15.6391C9.87152 15.8407 10.5169 15.9565 11.3339 15.9868V11.7015H11.2629C10.3452 11.6712 9.55181 11.4798 8.88277 11.1271C8.05388 10.6937 7.63943 10.0916 7.63943 9.32074C7.63943 8.63045 8.01835 7.98048 8.7762 7.37081C9.47484 6.80649 10.3274 6.39584 11.3339 6.13888V4.68777C11.3339 4.49126 11.4109 4.32751 11.5648 4.1965C11.7247 4.0655 11.9201 4 12.151 4C12.6602 4 12.9147 4.21414 12.9147 4.64242Z" fill="white"/>
</svg>`;
const REGULAR_BLUE_DOLLAR_SVG = (
isAriaLabel,
className
) => `<svg class="${className}" ${
isAriaLabel ? `aria-label="${VERIFIED_ACCOUNT_ARIA_LABEL}"` : ""
} ${
isAriaLabel ? `data-eight-dollars-original-classes="${className}"` : ""
} style="margin-left: 2px; max-width: 20px; max-height: 20px;" width="24px" height="24px" viewBox="0 0 24 24" data-eight-dollars-status="blueVerified" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<path d="M22.25,12 C22.25,10.57 21.37,9.33 20.06,8.66 C20.52,7.27 20.26,5.76 19.25,4.75 C18.24,3.74 16.73,3.48 15.34,3.94 C14.68,2.63 13.43,1.75 12,1.75 C10.57,1.75 9.33,2.63 8.67,3.94 C7.27,3.48 5.76,3.74 4.75,4.75 C3.74,5.76 3.49,7.27 3.95,8.66 C2.64,9.33 1.75,10.57 1.75,12 C1.75,13.43 2.64,14.67 3.95,15.34 C3.49,16.73 3.74,18.24 4.75,19.25 C5.76,20.26 7.27,20.51 8.66,20.06 C9.33,21.37 10.57,22.25 12,22.25 C13.43,22.25 14.68,21.37 15.34,20.06 C16.73,20.51 18.24,20.26 19.25,19.25 C20.26,18.24 20.52,16.73 20.06,15.34 C21.37,14.67 22.25,13.43 22.25,12 Z" id="Path" fill="#1D9BF0" fill-rule="nonzero"></path>
<path d="M13.8953012,9.17414679 C13.7123981,8.90893727 13.4677652,8.7146027 13.1614025,8.5911431 C12.8550397,8.46768349 12.532673,8.40595369 12.1943022,8.40595369 C11.9931088,8.40595369 11.7987742,8.42881658 11.6112985,8.47454236 C11.4238228,8.52026814 11.2546374,8.59342938 11.1037424,8.6940261 C10.9528473,8.79462282 10.831674,8.92494129 10.7402224,9.08498152 C10.6487709,9.24502175 10.6030451,9.43478374 10.6030451,9.65426748 C10.6030451,9.9834931 10.7173595,10.2349849 10.9459884,10.4087429 C11.1746173,10.5825008 11.4581172,10.7333959 11.7964879,10.8614281 C12.1348587,10.9894603 12.5052375,11.1129199 12.9076244,11.2318069 C13.3100113,11.3506939 13.6803901,11.5153067 14.0187608,11.7256453 C14.3571316,11.9359839 14.6406315,12.2149112 14.8692604,12.5624271 C15.0978893,12.909943 15.2122037,13.3717734 15.2122037,13.9479182 C15.2122037,14.4691921 15.1161796,14.9241636 14.9241313,15.3128328 C14.732083,15.7015019 14.4737324,16.0238686 14.1490793,16.279933 C13.8244263,16.5359974 13.4471886,16.7280457 13.0173663,16.8560778 C12.5875439,16.98411 12.1348587,17.0481261 11.6593106,17.0481261 C11.0557303,17.0481261 10.4750129,16.9475294 9.91715837,16.746336 C9.35930385,16.5451425 8.87918316,16.2067718 8.4767963,15.7312237 L9.77998103,14.4691921 C9.99031962,14.7892726 10.2669606,15.0384781 10.6099039,15.2168086 C10.9528473,15.3951392 11.3163672,15.4843044 11.7004638,15.4843044 C11.9016572,15.4843044 12.1028507,15.456869 12.3040441,15.401998 C12.5052375,15.3471271 12.6881406,15.2648207 12.8527535,15.1550788 C13.0173663,15.045337 13.149971,14.9058733 13.2505677,14.7366879 C13.3511645,14.5675026 13.4014628,14.373168 13.4014628,14.1536842 C13.4014628,13.7970232 13.2871484,13.5226685 13.0585195,13.3306202 C12.8298906,13.1385719 12.5463907,12.9762454 12.20802,12.8436406 C11.8696492,12.7110359 11.4992704,12.58529 11.0968835,12.466403 C10.6944966,12.3475159 10.3241178,12.1851894 9.98574704,11.9794234 C9.64737627,11.7736574 9.36387643,11.4993027 9.13524753,11.1563594 C8.90661863,10.813416 8.79230418,10.3538719 8.79230418,9.77772709 C8.79230418,9.27474351 8.89518719,8.8403486 9.1009532,8.47454236 C9.30671921,8.10873612 9.57650131,7.80465968 9.9102995,7.56231304 C10.2440977,7.31996641 10.625908,7.13934958 11.0557303,7.02046255 C11.4855526,6.90157552 11.9245201,6.84213201 12.3726328,6.84213201 C12.8847615,6.84213201 13.3808862,6.91986583 13.8610069,7.07533349 C14.3411276,7.23080114 14.7732362,7.48686551 15.1573328,7.84352659 L13.8953012,9.17414679 Z" id="Path" fill="#FFFFFF" fill-rule="nonzero"></path>
<polygon id="Path" fill="#FFFFFF" fill-rule="nonzero" points="12.4306693 5.98530124 12.4306693 18.0146988 11.375459 18.0146988 11.375459 5.98530124"></polygon>
</svg>`;
const REGULAR_BLUE_CHECK_SVG = (isAriaLabel, className) =>
`<svg viewBox="0 0 24 24" ${
isAriaLabel ? `aria-label="${VERIFIED_ACCOUNT_ARIA_LABEL}"` : ""
} ${
isAriaLabel ? `data-eight-dollars-original-classes="${className}"` : ""
} style="margin-left: 2px; max-width: 20px; max-height: 20px;" width="24px" height="24px" role="img" data-eight-dollars-status="verified" class="${className}"><g><path d="M22.25 12c0-1.43-.88-2.67-2.19-3.34.46-1.39.2-2.9-.81-3.91s-2.52-1.27-3.91-.81c-.66-1.31-1.91-2.19-3.34-2.19s-2.67.88-3.33 2.19c-1.4-.46-2.91-.2-3.92.81s-1.26 2.52-.8 3.91c-1.31.67-2.2 1.91-2.2 3.34s.89 2.67 2.2 3.34c-.46 1.39-.21 2.9.8 3.91s2.52 1.26 3.91.81c.67 1.31 1.91 2.19 3.34 2.19s2.68-.88 3.34-2.19c1.39.45 2.9.2 3.91-.81s1.27-2.52.81-3.91c1.31-.67 2.19-1.91 2.19-3.34zm-11.71 4.2L6.8 12.46l1.41-1.42 2.26 2.26 4.8-5.23 1.47 1.36-6.2 6.76z" fill="#1D9BF0"></path></g></svg>`;
function getOriginalClasses(elm) {
if (elm.dataset.eightDollarsOriginalClasses) {
return elm.dataset.eightDollarsOriginalClasses;
}
return [...elm.classList].join(" ");
}
function changeVerified(prependHTML, elm, isSmall, isIndeterminate) {
if (elm.dataset.eightDollarsStatus === "verified") {
// already replaced this element
return;
}
// hide if believe there's bad data
if (isIndeterminate) {
elm.style.display = "none";
elm.setAttribute("data-eight-dollars-status", "verified");
return;
}
const small = REGULAR_BLUE_CHECK_SVG(true, getOriginalClasses(elm));
const smallInnerElement = REGULAR_BLUE_CHECK_SVG(
false,
getOriginalClasses(elm)
);
const big = `
<span style='margin-left: 0.25rem; display: flex; flex-direction: row; align-items: center;${
TEXT_ENABLE_BORDER
? ` border-radius: 120px; border: 1px solid #536471;`
: ``
} padding: 0.1rem 0.4rem 0.1rem 0.2rem; gap: ${
TEXT_VERIFIED_LABEL ? 0.2 : 0
}rem;' aria-label="${VERIFIED_ACCOUNT_ARIA_LABEL}" data-eight-dollars-status="verified" data-eight-dollars-original-classes="${getOriginalClasses(
elm
)}">
${smallInnerElement}
<p style=' font-size: 0.8rem; margin: 0; font-weight: 600;'>${TEXT_VERIFIED_LABEL}</p>
</span>`;
try {
if (prependHTML !== "") {
// Ideally, we wouldn't mutate the parent element, because those 2 styles won't be managed by us further on.
// That is, if the `aria-label`-selected element changes, the parent styles won't be properly updated.
// This approach is tolerable, because it's unlikely that a `aria-label`-selected element changes from a
// "React text node(s) + React element node" sibling configuration to a "single React element node" sibling configuration.
elm.style.display = "inline-flex";
elm.style.alignItems = "center";
}
if (isSmall || !TEXT_ENABLED) {
elm.innerHTML = `${prependHTML}${small}`;
} else {
elm.innerHTML = `${prependHTML}${big}`;
}
} catch (e) {
console.error("error changing verified", e);
}
}
function changeBlueVerified(prependHTML, elm, isSmall) {
if (elm.dataset.eightDollarsStatus === "blueVerified") {
// already replaced this element
return;
}
if (REMOVE_TWITTER_BLUE_VERIFICATION) {
elm.style.display = "none";
elm.setAttribute("data-eight-dollars-status", "blueVerified");
return;
}
const small = MEME_MODE
? `${COMIC_SANS_BLUE_DOLLAR_SVG(true, getOriginalClasses(elm))}`
: `${REGULAR_BLUE_DOLLAR_SVG(true, getOriginalClasses(elm))}`;
const smallInnerElement = MEME_MODE
? `${COMIC_SANS_BLUE_DOLLAR_SVG(false, getOriginalClasses(elm))}`
: `${REGULAR_BLUE_DOLLAR_SVG(false, getOriginalClasses(elm))}`;
const big = `
<span style='margin-left: 0.25rem; display: flex; flex-direction: row; align-items: center;${
TEXT_ENABLE_BORDER
? ` border-radius: 120px; border: 1px solid #536471;`
: ``
} padding: 0.1rem 0.4rem 0.1rem 0.2rem; gap: ${
TEXT_TWITTER_BLUE_LABEL ? 0.2 : 0
}rem;' aria-label="${VERIFIED_ACCOUNT_ARIA_LABEL}" data-eight-dollars-status="blueVerified" data-eight-dollars-original-classes="${getOriginalClasses(
elm
)}">
${smallInnerElement}
<p style=' font-size: 0.8rem; margin: 0; font-weight: 600;'>${TEXT_TWITTER_BLUE_LABEL}</p>
</span>`;
try {
if (prependHTML !== "") {
// Ideally, we wouldn't mutate the parent element, because those 2 styles won't be managed by us further on.
// That is, if the `aria-label`-selected element changes, the parent styles won't be properly updated.
// This approach is tolerable, because it's unlikely that a `aria-label`-selected element changes from a
// "React text node(s) + React element node" sibling configuration to a "single React element node" sibling configuration.
elm.style.display = "inline-flex";
elm.style.alignItems = "center";
}
if (isSmall || !TEXT_ENABLED) {
elm.innerHTML = `${prependHTML}${small}`;
} else {
elm.innerHTML = `${prependHTML}${big}`;
}
} catch (e) {
console.error("error changing blue verified", e);
}
}
function querySelectorAllIncludingMe(node, selector) {
if (node.matches(selector)) {
return [node];
}
return [...node.querySelectorAll(selector)];
}
const trackingTweets = new Set();
const trackingHovercards = new Set();
const trackingUsercells = new Set();
const trackingConversation = new Set();
const trackingDmDrawHeader = new Set();
const trackingUserName = new Set();
const trackingHeadings = new Set();
function collectAndTrackElements(node) {
const tweets = querySelectorAllIncludingMe(node, '[data-testid="tweet"]');
for (const tweet of tweets) {
trackingTweets.add(tweet);
}
const hovercards = querySelectorAllIncludingMe(
node,
'[data-testid="HoverCard"]'
);
for (const hovercard of hovercards) {
trackingHovercards.add(hovercard);
}
const usercells = querySelectorAllIncludingMe(
node,
'[data-testid="UserCell"]'
);
for (const usercell of usercells) {
trackingUsercells.add(usercell);
}
const conversations = querySelectorAllIncludingMe(
node,
'[data-testid="conversation"]'
);
for (const conversation of conversations) {
trackingConversation.add(conversation);
}
const dmDrawHeader = querySelectorAllIncludingMe(
node,
'[data-testid="DMDrawerHeader"]'
);
for (const dm of dmDrawHeader) {
trackingDmDrawHeader.add(dm);
}
const userName = querySelectorAllIncludingMe(
node,
'[data-testid="UserName"]'
);
for (const name of userName) {
trackingUserName.add(name);
}
const headings = querySelectorAllIncludingMe(
node,
'[data-testid="primaryColumn"] [role="heading"][aria-level="2"][dir="ltr"]'
);
for (const heading of headings) {
trackingHeadings.add(heading);
}
}
function handleProfileModification(containerElement, isBig) {
const ltr = containerElement.querySelectorAll('[dir="ltr"]');
const checkmarkContainer = ltr[0];
const spans = checkmarkContainer.querySelectorAll("span");
const checkmarkSpan = spans[spans.length - 1];
const protectd = checkmarkSpan.querySelectorAll('[aria-label="Protected account"]')[0]
if (!checkmarkSpan || protectd) {
console.error("no checkmark span found");
return;
}
let hasVerifiedIcon = checkmarkSpan.children.length > 0;
const handleContainer = ltr[1];
const handle = handleContainer.innerText;
if (!handle) {
return;
}
const parsedHandle = handle.replace("@", "").toLowerCase();
const isLegacyVerifed = VERIFIED_USERS.has(parsedHandle);
if (isLegacyVerifed) {
changeVerified("", checkmarkSpan, isBig);
} else if (hasVerifiedIcon) {
changeBlueVerified("", checkmarkSpan, isBig);
}
if (DEBUG) {
if (isLegacyVerifed) {
containerElement.style = "border: 1px solid green;";
} else if (hasVerifiedIcon) {
containerElement.style = "border: 1px solid red;";
}
// add a plain text node to the tweet
containerElement.appendChild(
document.createTextNode(
`isLegacyVerified: ${isLegacyVerifed.toString()}\n verifiedIcon: ${hasVerifiedIcon.toString()}\n`
)
);
}
}
function waitForElm(selector) {
return new Promise((resolve) => {
if (document.querySelector(selector)) {
return resolve(document.querySelector(selector));
}
const observer = new MutationObserver((mutations) => {
if (document.querySelector(selector)) {
resolve(document.querySelector(selector));
observer.disconnect();
}
});
observer.observe(document.body, {
childList: true,
subtree: true,
});
});
}
async function handleHeadingModification(containerElement, isBig) {
const profile = await waitForElm('[data-testid="UserName"]');
if (!profile) {
console.error("no profile found");
return;
}
const ltr = profile.querySelectorAll('[dir="ltr"]');
const handleContainer = ltr[1];
const handle = handleContainer.innerText;
if (!handle) {
return;
}
const parsedHandle = handle.replace("@", "").toLowerCase();
const isLegacyVerifed = VERIFIED_USERS.has(parsedHandle);
const spans = containerElement.querySelectorAll("span");
const checkmarkSpan = spans[spans.length - 1];
const protectd = checkmarkSpan.querySelectorAll('[aria-label="Protected account"]')[0]
if (!checkmarkSpan || protectd) {
console.error("no checkmark span found");
return;
}
let hasVerifiedIcon = checkmarkSpan.children.length > 0;
if (isLegacyVerifed) {
changeVerified("", checkmarkSpan, isBig);
} else if (hasVerifiedIcon) {
changeBlueVerified("", checkmarkSpan, isBig);
}
if (DEBUG) {
if (isLegacyVerifed) {
containerElement.style = "border: 1px solid green;";
} else if (hasVerifiedIcon) {
containerElement.style = "border: 1px solid red;";
}
// add a plain text node to the tweet
containerElement.appendChild(
document.createTextNode(
`isLegacyVerified: ${isLegacyVerifed.toString()}\n verifiedIcon: ${hasVerifiedIcon.toString()}\n`
)
);
}
}
function handleModification(
containerElement,
checkmarkIndex,
handleIndex,
isBig
) {
const ltr = containerElement.querySelectorAll('[dir="ltr"]');
const checkmarkContainer = ltr[checkmarkIndex];
const checkmarkSpan = checkmarkContainer.children?.[0];
const protectd = checkmarkSpan.querySelectorAll('[aria-label="Protected account"]')[0]
if (!checkmarkSpan || protectd) {
return;
}
let hasVerifiedIcon = checkmarkSpan.children.length > 0;
const handleContainer = ltr[handleIndex];
const handle = handleContainer.innerText;
if (!handle) {
return;
}
const parsedHandle = handle.replace("@", "").toLowerCase();
const isLegacyVerifed = VERIFIED_USERS.has(parsedHandle);
if (isLegacyVerifed) {
changeVerified("", checkmarkSpan, isBig);
} else if (hasVerifiedIcon) {
changeBlueVerified("", checkmarkSpan, isBig);
}
if (DEBUG) {
if (isLegacyVerifed) {
containerElement.style = "border: 1px solid green;";
} else if (hasVerifiedIcon) {
containerElement.style = "border: 1px solid red;";
}
// add a plain text node to the tweet
containerElement.appendChild(
document.createTextNode(
`isLegacyVerified: ${isLegacyVerifed.toString()}\n verifiedIcon: ${hasVerifiedIcon.toString()}\n`
)
);
}
}
async function main() {
const observer = new MutationObserver(function (mutations, observer) {
try {
for (const mutation of mutations) {
if (mutation.type === "attributes") {
collectAndTrackElements(mutation.target);
}
for (const node of mutation.addedNodes) {
if (node.nodeType === 1) {
collectAndTrackElements(node);
}
}
}
for (const tweet of trackingTweets) {
if (!tweet.dataset.processed) {
// includes subtweets
const userNameContainer = tweet.querySelectorAll(
'[data-testid="User-Name"]'
);
for (const container of userNameContainer) {
handleModification(container, 1, 2, false);
}
tweet.dataset.processed = true;
}
}
for (const hovercard of trackingHovercards) {
if (!hovercard.dataset.processed) {
handleModification(hovercard, 2, 3, false);
hovercard.dataset.processed = true;
}
}
for (const usercell of trackingUsercells) {
if (!usercell.dataset.processed) {
handleModification(usercell, 1, 2, true);
usercell.dataset.processed = true;
}
}
for (const conversation of trackingConversation) {
if (!conversation.dataset.processed) {
handleModification(conversation, 1, 2, true);
conversation.dataset.processed = true;
}
}
for (const dm of trackingDmDrawHeader) {
if (!dm.dataset.processed) {
//handleModification(dm, 2, 2, true);
dm.dataset.processed = true;
}
}
for (const name of trackingUserName) {
if (!name.dataset.processed) {
handleProfileModification(name);
name.dataset.processed = true;
}
}
for (const heading of trackingHeadings) {
if (!heading.dataset.processed) {
handleHeadingModification(heading);
heading.dataset.processed = true;
}
}
} catch (error) {
console.log("uncaught mutation error", error);
}
});
observer.observe(document, {
childList: true,
subtree: true,
attributes: true,
});
}
main();