forked from pseudonymous/better-better-booru
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbetter-better-booru.user.js
9526 lines (7728 loc) · 407 KB
/
better-better-booru.user.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
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// ==UserScript==
// @name better_better_booru
// @namespace https://greasyfork.org/scripts/3575-better-better-booru
// @author otani, modified by Jawertae, A Pseudonymous Coder & Moebius Strip.
// @description Several changes to make Danbooru much better. Including the viewing of hidden/censored images on non-upgraded accounts and more.
// @version 7.2.5
// @updateURL https://greasyfork.org/scripts/3575-better-better-booru/code/better_better_booru.meta.js
// @downloadURL https://greasyfork.org/scripts/3575-better-better-booru/code/better_better_booru.user.js
// @match *://*.donmai.us/*
// @run-at document-end
// @grant none
// @icon 
// ==/UserScript==
// Have a nice day. - A Pseudonymous Coder
function bbbScript() { // This is needed to make this script work in Chrome.
/*
* NOTE: You no longer need to edit this script to change settings!
* Use the "BBB Settings" button in the menu instead.
*/
// If Danbooru's JS isn't available, assume we're somewhere this script isn't needed and stop.
if (typeof(Danbooru) === "undefined")
return;
/* Helper Prototypes */
// Don't get hoisted so they should be declared at the top to simplify things.
String.prototype.bbbSpacePad = function() {
// Add a leading and trailing space.
return (this.length ? " " + this + " " : "");
};
String.prototype.bbbSpaceClean = function() {
// Remove leading, trailing, and multiple spaces.
return this.replace(/\s+/g, " ").replace(/^\s|\s$/g, "");
};
String.prototype.bbbTagClean = function() {
// Remove extra commas along with leading, trailing, and multiple spaces.
return this.replace(/[\s,]*(%\))\s*|\s*([~-]*\(%)[\s,]*/g, " $& ").replace(/[\s,]*,[\s,]*/g, ", ").replace(/[\s,]+$|^[\s,]+/g, "").replace(/\s+/g, " ");
};
String.prototype.bbbHash = function() {
// Turn a string into a hash using the current Danbooru hash method.
var hash = 5381;
var i = this.length;
while(i)
hash = (hash * 33) ^ this.charCodeAt(--i);
return hash >>> 0;
};
Element.prototype.bbbGetPadding = function() {
// Get all the padding measurements of an element including the total width and height.
if (window.getComputedStyle) {
var computed = window.getComputedStyle(this, null);
var paddingLeft = parseFloat(computed.paddingLeft);
var paddingRight = parseFloat(computed.paddingRight);
var paddingTop = parseFloat(computed.paddingTop);
var paddingBottom = parseFloat(computed.paddingBottom);
var paddingHeight = paddingTop + paddingBottom;
var paddingWidth = paddingLeft + paddingRight;
return {width: paddingWidth, height: paddingHeight, top: paddingTop, bottom: paddingBottom, left: paddingLeft, right: paddingRight};
}
};
Element.prototype.bbbHasClass = function() {
// Test an element for one or more collections of classes.
var classList = this.classList;
for (var i = 0, il = arguments.length; i < il; i++) {
var classes = arguments[i].bbbSpaceClean();
if (!classes)
continue;
var classArray = classes.split(" ");
var hasClass = true;
for (var j = 0, jl = classArray.length; j < jl; j++) {
if (!classList.contains(classArray[j])) {
hasClass = false;
break;
}
}
if (hasClass)
return true;
}
return false;
};
Element.prototype.bbbAddClass = function(classString) {
// Add one or more classes to an element.
var classes = classString.bbbSpaceClean();
if (!classes)
return;
var classList = this.classList;
var classArray = classes.split(" ");
for (var i = 0, il = classArray.length; i < il; i++)
classList.add(classArray[i]);
};
Element.prototype.bbbRemoveClass = function(classString) {
// Remove one or more classes from an element.
var classes = classString.bbbSpaceClean();
if (!classes)
return;
var classList = this.classList;
var classArray = classes.split(" ");
for (var i = 0, il = classArray.length; i < il; i++)
classList.remove(classArray[i]);
};
Element.prototype.bbbWatchNodes = function(func) {
// Watch for new nodes.
var observer = window.MutationObserver || window.WebKitMutationObserver;
if (observer) {
observer = new observer(func);
observer.observe(this, {childList: true, subtree: true});
}
else
this.addEventListener("DOMNodeInserted", func, false);
};
Element.prototype.bbbOverrideClick = function(func) {
// Override Danbooru's click event listeners by capturing clicks on the parent node and stopping them.
var target = this;
var wrapperFunc = function(event) {
if (event.target !== target || event.button !== 0)
return;
func(event);
event.stopPropagation();
};
target.parentNode.addEventListener("click", wrapperFunc, true);
};
Storage.prototype.bbbSetItem = function(key, value) {
// Store a value in storage and warn if it is full.
try {
this.setItem(key, value);
}
catch (error) {
if (error.code === 22 || error.code === 1014) {
if (this === localStorage) {
if (!bbb.flags.local_storage_full) {
if (localStorage.length > 2000) {
// Try clearing out autocomplete if that appears to be the problem.
cleanLocalStorage("autocomplete");
try {
localStorage.setItem(key, value);
}
catch (localError) {
bbb.flags.local_storage_full = true;
}
}
else
bbb.flags.local_storage_full = true;
// Store the local storage value until it can be retried.
if (bbb.flags.local_storage_full) {
bbb.local_storage_queue = {};
bbb.local_storage_queue[key] = value;
localStorageDialog();
}
}
else {
// Temporarily store additional local storage values until they can be retried.
if (sessionStorage.getItem("bbb_local_storage_queue")) {
var sessLocal = JSON.parse(sessionStorage.getItem("bbb_local_storage_queue"));
sessLocal[key] = value;
sessionStorage.bbbSetItem("bbb_local_storage_queue", JSON.stringify(sessLocal));
}
else
bbb.local_storage_queue[key] = value;
}
}
else {
// Keep only a few values in session storage.
for (var i = sessionStorage.length - 1; i >= 0; i--) {
var keyName = sessionStorage.key(i);
if (keyName !== "bbb_endless_default" && keyName !== "bbb_quick_search")
sessionStorage.removeItem(keyName);
}
try {
sessionStorage.setItem(key, value);
}
catch (sessionError) {
bbbNotice("Your settings/data could not be saved/updated. The browser's session storage is full.", -1);
}
}
}
else
bbbNotice("Unexpected error while attempting to save/update settings. (Error: " + error.message + ")", -1);
}
};
/* Global Variables */
var bbb = { // Container for script info.
blacklist: {
entries: [],
match_list: {},
smart_view_target: undefined
},
cache: { // Thumbnail info cache.
current: {
history: [],
names: {}
},
save_enabled: false,
stored: {}
},
custom_tag: {
searches: [],
style_list: {}
},
dialog: {
queue: []
},
drag_scroll: {
lastX: undefined,
lastY: undefined,
moved: false,
target: undefined
},
el: { // Script elements.
menu: {} // Menu elements.
},
endless: {
append_page: false,
enabled: false,
fill_first_page: false,
last_paginator: undefined,
new_paginator: undefined,
no_thumb_count: 0,
pages: [],
paused: false,
posts: {}
},
fixed_paginator_space: 0,
fixed_sidebar: {
content: undefined,
left: undefined,
sidebar: undefined,
top: undefined
},
flags: {},
hotkeys: {
other: { // Hotkeys for misc locations.
66: {func: openMenu} // B
},
post: { // Post hotkeys.
49: {func: resizeHotkey, custom_handler: true}, // 1
50: {func: resizeHotkey, custom_handler: true}, // 2
51: {func: resizeHotkey, custom_handler: true}, // 3
52: {func: resizeHotkey, custom_handler: true}, // 4
66: {func: openMenu}, // B
86: {func: swapPost} // V
}
},
post: { // Post content info and status.
info: {}, // Post information object.
resize: {
mode: "none",
ratio: 1
},
swapped: false // Whether the post content has been changed between the original and sample versions.
},
options: { // Setting options and data.
bbb_version: "7.2.5",
alternate_image_swap: newOption("checkbox", false, "Alternate Image Swap", "Switch between the sample and original image by clicking the image. <tiphead>Note</tiphead>Notes can be toggled by using the link in the sidebar options section."),
arrow_nav: newOption("checkbox", false, "Arrow Navigation", "Allow the use of the left and right arrow keys to navigate pages. <tiphead>Note</tiphead>This option has no effect on individual posts."),
autohide_sidebar: newOption("dropdown", "none", "Auto-hide Sidebar", "Hide the sidebar for posts, favorites listings, and/or searches until the mouse comes close to the left side of the window or the sidebar gains focus.<tiphead>Tips</tiphead>By using Danbooru's hotkey for the letter \"Q\" to place focus on the search box, you can unhide the sidebar.<br><br>Use the \"thumbnail count\" option to get the most out of this feature on search listings.", {txtOptions:["Disabled:none", "Favorites:favorites", "Posts:post", "Searches:search", "Favorites & Posts:favorites post", "Favorites & Searches:favorites search", "Posts & Searches:post search", "All:favorites post search"]}),
autoscroll_post: newOption("dropdown", "none", "Auto-scroll Post", "Automatically scroll a post to a particular point. <tipdesc>Below Header:</tipdesc> Scroll the window down until the header is no longer visible or scrolling is no longer possible. <tipdesc>Post Content:</tipdesc> Position the post content as close as possible to the left and top edges of the window viewport when initially loading a post. Using this option will also scroll past any notices above the content.", {txtOptions:["Disabled:none", "Below Header:header", "Post Content:post"]}),
blacklist_add_bars: newOption("checkbox", false, "Additional Bars", "Add a blacklist bar to the comment search listing and individually linked comments so that blacklist entries can be toggled as needed."),
blacklist_highlight_color: newOption("text", "#CCCCCC", "Highlight Color", "When using highlighting for \"thumbnail marking\", you may set the color here. <tiphead>Notes</tiphead>Leaving this field blank will result in the default color being used. <br><br>For easy color selection, use one of the many free tools on the internet like <a target=\"_blank\" href=\"http://www.quackit.com/css/css_color_codes.cfm\">this one</a>. Hex RGB color codes (#000000, #FFFFFF, etc.) are the recommended values."),
blacklist_thumb_controls: newOption("checkbox", false, "Thumbnail Controls", "Allow control over individual blacklisted thumbnails and access to blacklist toggle links from blacklisted thumbnails. <tiphead>Directions</tiphead>For blacklisted thumbnails that have been revealed, hovering over them will reveal a clickable \"X\" icon that can hide them again. <br><br>If using \"hidden\" or \"replaced\" for the \"post display\" option, clicking on the area of a blacklisted thumbnail will pop up a menu that displays what blacklist entries it matches. Clicking the thumbnail area a second time while that menu is open will reveal that single thumbnail. <br><br>The menu that pops up on the first click also allows for toggling any listed blacklist entry for the entire page and navigating to the post without revealing its thumbnail. <tiphead>Note</tiphead>Toggling blacklist entries will have no effect on posts that have been changed via their individual controls."),
blacklist_post_display: newOption("dropdown", "disabled", "Post Display", "Set how the display of blacklisted posts in thumbnail listings and the comments section is handled. <tipdesc>Removed:</tipdesc> Posts and the space they take up are completely removed. <tipdesc>Hidden:</tipdesc> Post space is preserved, but thumbnails are hidden. <tipdesc>Replaced:</tipdesc> Thumbnails are replaced by \"blacklisted\" thumbnail placeholders.", {txtOptions:["Disabled:disabled", "Removed:removed", "Hidden:hidden", "Replaced:replaced"]}),
blacklist_smart_view: newOption("checkbox", false, "Smart View", "When navigating to a blacklisted post by using its thumbnail, if the thumbnail has already been revealed, the post content will temporarily be exempt from any blacklist checks for 1 minute and be immediately visible. <tiphead>Note</tiphead>Thumbnails in the parent/child notices of posts with exempt content will still be affected by the blacklist."),
blacklist_session_toggle: newOption("checkbox", false, "Session Toggle", "When toggling an individual blacklist entry on and off, the mode it's toggled to will persist across other pages in the same browsing session until it ends.<tiphead>Note</tiphead>For blacklists with many entries, this option can cause unexpected behavior (ex: getting logged out) if too many entries are toggled off at the same time."),
blacklist_thumb_mark: newOption("dropdown", "none", "Thumbnail Marking", "Mark the thumbnails of blacklisted posts that have been revealed to make them easier to distinguish from other thumbnails. <tipdesc>Highlight:</tipdesc> Change the background color of blacklisted thumbnails. <tipdesc>Icon Overlay:</tipdesc> Add an icon to the lower right corner of blacklisted thumbnails.", {txtOptions:["Disabled:none", "Highlight:highlight", "Icon Overlay:icon"]}),
border_spacing: newOption("dropdown", 0, "Border Spacing", "Set the amount of blank space between a border and thumbnail and between a custom tag border and status border. <tiphead>Note</tiphead>Even when set to 0, status borders and custom tag borders will always have a minimum value of 1 between them. <tiphead>Tip</tiphead>Use this option if you often have trouble distinguishing a border from the thumbnail image.", {txtOptions:["0 (Default):0", "1:1", "2:2", "3:3"]}),
border_width: newOption("dropdown", 2, "Border Width", "Set the width of thumbnail borders.", {txtOptions:["1:1", "2 (Default):2", "3:3", "4:4", "5:5"]}),
bypass_api: newOption("checkbox", false, "Automatic API Bypass", "When logged out and API only features are enabled, do not warn about needing to be logged in. Instead, automatically bypass those features."),
clean_links: newOption("checkbox", false, "Clean Links", "Remove the extra information after the post ID in thumbnail links.<tiphead>Note</tiphead>Enabling this option will disable Danbooru's search navigation and active pool/favorite group detection for posts."),
collapse_sidebar: newOption("checkbox", false, "Collapsible Sidebar", "Allow sections in the sidebar to be expanded and collapsed via clicking their header titles.<tiphead>Note</tiphead>Sections can be set to default to expanded or collapsed by right clicking their titles."),
comment_score: newOption("checkbox", false, "Comment Scores", "Make comment scores visible by adding them as direct links to their respective comments."),
custom_status_borders: newOption("checkbox", false, "Custom Status Borders", "Override Danbooru's thumbnail borders for deleted, flagged, pending, parent, and child images."),
custom_tag_borders: newOption("checkbox", true, "Custom Tag Borders", "Add thumbnail borders to posts with specific tags."),
direct_downloads: newOption("checkbox", false, "Direct Downloads", "Allow download managers to download the posts displayed in the favorites, search, pool, popular, and favorite group listings. <tiphead>Note</tiphead>Posts filtered out by the blacklist or quick search will not provide direct downloads until the blacklist entry or quick search affecting them is disabled."),
disable_embedded_notes: newOption("checkbox", false, "Disable Embedded Notes", "Force posts with embedded notes to display with the original note styling. <tiphead>Notes</tiphead>While notes will display with the original styling, the actual post settings will still have embedded notes set to enabled. <br><br>Due to the actual settings, users that may wish to edit notes will have to edit the notes with the embedded note styling so that nothing ends up breaking in unexpected ways. When toggling translation mode or opening the edit note dialog box, the notes will automatically revert back to the original embedded notes until the page is reloaded. <br><br>Note resizing and moving will be allowed without the reversion to embedded notes since this ability is sometimes necessary for badly positioned notes. Any note resizing or moving done as a part of intended note editing should be done <b>after</b> triggering the embedded note reversion since any changes before it will be lost."),
enable_status_message: newOption("checkbox", true, "Enable Status Message", "When requesting information from Danbooru, display the request status in the lower right corner."),
endless_default: newOption("dropdown", "disabled", "Default", "Enable endless pages on the favorites, search, pool, notes, and favorite group listings. <tipdesc>Off:</tipdesc> Start up with all features off. <tipdesc>On:</tipdesc> Start up with all features on.<tipdesc>Paused:</tipdesc> Start up with all features on, but do not append new pages until the \"load more\" button is clicked. <tiphead>Note</tiphead>When not set to disabled, endless pages can be toggled between off and on/paused by using the \"E\" hotkey or the \"endless\" link next to the \"listing\" link in the page submenu. <tiphead>Tip</tiphead>The \"new tab/window\" and \"fixed paginator\" options can provide additional customization for endless pages.", {txtOptions:["Disabled:disabled", "Off:off", "On:on", "Paused:paused"]}),
endless_fill: newOption("checkbox", false, "Fill Pages", "When appending pages with missing thumbnails caused by hidden posts or removed duplicate posts, retrieve thumbnails from the following pages and add them to the new page until the desired number of thumbnails is reached. <tiphead>Note</tiphead>If using page separators, the displayed page number for appended pages composed of thumbnails from multiple Danbooru pages will be replaced by a range consisting of the first and last pages from which thumbnails were retrieved."),
endless_pause_interval: newOption("dropdown", 0, "Pause Interval", "Pause endless pages each time the number of pages reaches a multiple of the selected amount.", {txtOptions:["Disabled:0"], numRange:[1,100]}),
endless_preload: newOption("checkbox", false, "Preload Next Page", "Start loading the next page as soon as possible.<tiphead>Note</tiphead>A preloaded page will not be appended until the scroll limit is reached."),
endless_remove_dup: newOption("checkbox", false, "Remove Duplicates", "When appending new pages, remove posts that already exist in the listing from the new page.<tiphead>Note</tiphead>Duplicate posts are caused by the addition of new posts to the beginning of a listing or changes to the order of the posts."),
endless_scroll_limit: newOption("dropdown", 500, "Scroll Limit", "Set the minimum amount of pixels that the window can have left to vertically scroll before it starts appending the next page.", {numList:[0,50,100,150,200,250,300,350,400,450,500,550,600,650,700,750,800,850,900,950,1000,1050,1100,1150,1200,1250,1300,1350,1400,1450,1500]}),
endless_separator: newOption("dropdown", "divider", "Page Separator", "Distinguish pages from each other by marking them with a separator.<tipdesc>Marker:</tipdesc> Place a thumbnail sized marker before the first thumbnail of each page.<tipdesc>Divider:</tipdesc> Completely separate pages by placing a horizontal line between them.", {txtOptions:["None:none", "Marker:marker", "Divider:divider"]}),
endless_session_toggle: newOption("checkbox", false, "Session Toggle", "When toggling endless pages on and off, the mode it's toggled to will override the default and persist across other pages in the same browsing session for that tab until it ends."),
fixed_paginator: newOption("dropdown", "disabled", "Fixed Paginator", "Make the paginator always visible for the favorites, search, pool, notes, and favorite group listings by fixing it to the bottom of the window when it would normally start scrolling out of view. <tipdesc>Endless:</tipdesc> Only change the paginator during endless pages browsing. <tipdesc>Normal:</tipdesc> Only change the paginator during normal browsing. <tipdesc>Always:</tipdesc> Change the paginator during normal and endless pages browsing. <tiphead>Note</tiphead>Options labeled with \"minimal\" will also make the fixed paginator smaller by removing most of the blank space within it.", {txtOptions:["Disabled:disabled", "Endless:endless", "Endless (Minimal):endless minimal", "Normal:normal", "Normal (Minimal):normal minimal", "Always:endless normal", "Always (Minimal):endless normal minimal"]}),
fixed_sidebar: newOption("dropdown", "none", "Fixed Sidebar", "Make the sidebar never completely vertically scroll out of view for posts, favorites listings, and/or searches by fixing it to the top or bottom of the window when it would normally start scrolling out of view. <tiphead>Note</tiphead>The \"auto-hide sidebar\" option will override this option if both try to modify the same page. <tiphead>Tip</tiphead>Depending on the available height in the browser window and the Danbooru location being modified, the \"tag scrollbars\", \"collapsible sidebar\", and/or \"remove tag headers\" options may be needed for best results.", {txtOptions:["Disabled:none", "Favorites:favorites", "Posts:post", "Searches:search", "Favorites & Posts:favorites post", "Favorites & Searches:favorites search", "Posts & Searches:post search", "All:favorites post search"]}),
hide_ban_notice: newOption("checkbox", false, "Hide Ban Notice", "Hide the Danbooru ban notice."),
hide_comment_notice: newOption("checkbox", false, "Hide Comment Guide Notice", "Hide the Danbooru comment guide notice."),
hide_pool_notice: newOption("checkbox", false, "Hide Pool Guide Notice", "Hide the Danbooru pool guide notice."),
hide_sign_up_notice: newOption("checkbox", false, "Hide Sign Up Notice", "Hide the Danbooru account sign up notice."),
hide_tag_notice: newOption("checkbox", false, "Hide Tag Guide Notice", "Hide the Danbooru tag guide notice."),
hide_tos_notice: newOption("checkbox", false, "Hide TOS Notice", "Hide the Danbooru terms of service agreement notice."),
hide_upgrade_notice: newOption("checkbox", false, "Hide Upgrade Notice", "Hide the Danbooru upgrade account notice."),
hide_upload_notice: newOption("checkbox", false, "Hide Upload Guide Notice", "Hide the Danbooru upload guide notice."),
image_swap_mode: newOption("dropdown", "load", "Image Swap Mode", "Set how swapping between the sample and original image is done.<tipdesc>Load First:</tipdesc> Display the image being swapped in after it has finished downloading. <tipdesc>View While Loading:</tipdesc> Immediately display the image being swapped in while it is downloading.", {txtOptions:["Load First (Default):load", "View While Loading:view"]}),
search_tag_scrollbars: newOption("dropdown", 0, "Search Tag Scrollbars", "Limit the length of the sidebar tag list for the search listing by restricting it to a set height in pixels. When the list exceeds the set height, a scrollbar will be added to allow the rest of the list to be viewed.", {txtOptions:["Disabled:0"], numList:[50,100,150,200,250,300,350,400,450,500,550,600,650,700,750,800,850,900,950,1000,1050,1100,1150,1200,1250,1300,1350,1400,1450,1500]}),
load_sample_first: newOption("checkbox", true, "Load Sample First", "Load sample images first when viewing a post.<tiphead>Note</tiphead>When logged in, the account's \"default image width\" setting will override this option. This behavior can be changed with the \"override sample setting\" option under the preferences tab."),
manage_cookies: newOption("checkbox", false, "Manage Notice Cookies", "When using the \"hide upgrade notice\", \"hide sign up notice\", and/or \"hide TOS notice\" options, also create cookies to disable these notices at the server level.<tiphead>Tip</tiphead>Use this feature if the notices keep flashing on your screen before being removed."),
minimize_status_notices: newOption("checkbox", false, "Minimize Status Notices", "Hide the Danbooru deleted, banned, flagged, appealed, and pending notices. When you want to see a hidden notice, you can click the appropriate status link in the information section of the sidebar."),
move_save_search: newOption("checkbox", false, "Move Save Search", "Move the \"save this search\" button into the related section in the sidebar."),
override_blacklist: newOption("dropdown", "logged_out", "Override Blacklist", "Allow the \"blacklist\" setting to override the default blacklist for logged out users and/or account blacklist for logged in users. <tipdesc>Logged out:</tipdesc> Override the default blacklist for logged out users. <tipdesc>Always:</tipdesc> Override the default blacklist for logged out users and account blacklist for logged in users.", {txtOptions:["Disabled:disabled", "Logged out:logged_out", "Always:always"]}),
override_resize: newOption("checkbox", false, "Override Resize Setting", "Allow the \"resize post\" setting to override the account \"fit images to window\" setting when logged in."),
override_sample: newOption("checkbox", false, "Override Sample Setting", "Allow the \"load sample first\" setting to override the account \"default image width\" setting when logged in. <tiphead>Note</tiphead>When using this option, your Danbooru account settings should have \"default image width\" set to the corresponding value of the \"load sample first\" script setting. Not doing so will cause your browser to always download both the sample and original image. If you often change the \"load sample first\" setting, leaving your account to always load the sample/850px image first is your best option."),
page_counter: newOption("checkbox", false, "Page Counter", "Add a page counter and \"go to page #\" input field near the top of listing pages. <tiphead>Note</tiphead>The total number of pages will not be displayed if the pages are using the \"previous & next\" paging system or the total number of pages exceeds the maximum amount allowed by your user account level."),
post_drag_scroll: newOption("checkbox", false, "Post Drag Scrolling", "While holding down left click on a post's content, mouse movement can be used to scroll the whole page and reposition the content.<tiphead>Note</tiphead>This option is automatically disabled when translation mode is active."),
post_link_new_window: newOption("dropdown", "none", "New Tab/Window", "Force post links in the search, pool, popular, favorites, notes, and favorite group listings to open in a new tab/window. <tipdesc>Endless:</tipdesc> Only use new tabs/windows during endless pages browsing. <tipdesc>Normal:</tipdesc> Only use new tabs/windows during normal browsing. <tipdesc>Always:</tipdesc> Use new tabs/windows during normal and endless pages browsing. <tiphead>Notes</tiphead>When this option is active, holding down the control and shift keys while clicking a post link will open the post in the current tab/window.<br><br>Whether the post opens in a new tab or a new window depends upon your browser configuration. <tiphead>Tip</tiphead>This option can be useful as a safeguard to keep accidental left clicks from disrupting endless pages.", {txtOptions:["Disabled:disabled", "Endless:endless", "Normal:normal", "Always:endless normal"]}),
post_resize: newOption("checkbox", true, "Resize Post", "Shrink large post content to fit the browser window when initially loading a post.<tiphead>Note</tiphead>When logged in, the account's \"fit images to window\" setting will override this option. This behavior can be changed with the \"override resize setting\" option under the preferences tab."),
post_resize_mode: newOption("dropdown", "width", "Resize Mode", "Choose how to shrink large post content to fit the browser window when initially loading a post.", {txtOptions:["Width (Default):width", "Height:height", "Width & Height:all"]}),
post_tag_scrollbars: newOption("dropdown", 0, "Post Tag Scrollbars", "Limit the length of the sidebar tag lists for posts by restricting them to a set height in pixels. For lists that exceed the set height, a scrollbar will be added to allow the rest of the list to be viewed.<tiphead>Note</tiphead>When using \"remove tag headers\", this option will limit the overall length of the combined list.", {txtOptions:["Disabled:0"], numList:[50,100,150,200,250,300,350,400,450,500,550,600,650,700,750,800,850,900,950,1000,1050,1100,1150,1200,1250,1300,1350,1400,1450,1500]}),
post_tag_titles: newOption("checkbox", false, "Post Tag Titles", "Change the page titles for posts to a full list of the post tags."),
quick_search: newOption("dropdown", "disabled", "Quick Search", "Add a new search box to the upper right corner of the window viewport that allows searching through the current thumbnails for specific posts. <tipdesc>Fade:</tipdesc> Fade all posts that don't match in the thumbnail listing. <tipdesc>Remove:</tipdesc> Remove all posts that don't match from the thumbnail listing. <tiphead>Directions</tiphead>Please read the \"thumbnail matching rules\" section under the help tab for information about creating searches. <br><br>The search starts minimized in the upper right corner. Left clicking the main icon will open and close the search. Right clicking the main icon will completely reset the search. Holding down shift while left clicking the main icon will toggle an active search's pinned status. <br><br>While open, the search can be entered/updated in the search box and the pinned status can be toggled by clicking the pushpin icon. If no changes are made to an active search, submitting it a second time will reset the quick search. <tiphead>Notes</tiphead>Options labeled with \"pinned\" will make searches default to being pinned. <br><br>A pinned search will persist across other pages in the same browsing session for that tab until it ends or the search is unpinned. <br><br>When not set to disabled, the quick search can be opened by using the \"F\" hotkey. Additionally, an active search can be reset by using \"Shift + F\". Pressing \"Escape\" while the quick search is open will close it.", {txtOptions:["Disabled:disabled", "Fade:fade", "Fade (Pinned):fade pinned", "Remove:remove", "Remove (Pinned):remove pinned"]}),
remove_tag_headers: newOption("checkbox", false, "Remove Tag Headers", "Remove the \"copyrights\", \"characters\", and \"artist\" headers from the sidebar tag list."),
resize_link_style: newOption("dropdown", "full", "Resize Link Style", "Set how the resize links in the post sidebar options section will display. <tipdesc>Full:</tipdesc> Show the \"resize to window\", \"resize to window width\", and \"resize to window height\" links on separate lines. <tipdesc>Minimal:</tipdesc> Show the \"resize to window\" (W&H), \"resize to window width\" (W), and \"resize to window height\" (H) links on one line.", {txtOptions:["Full:full", "Minimal:minimal"]}),
script_blacklisted_tags: "",
search_add: newOption("dropdown", "disabled", "Search Add", "Modify the sidebar tag list by adding, removing, or replacing links in the sidebar tag list that modify the current search's tags. <tipdesc>Remove:</tipdesc> Remove any preexisting \"+\" and \"–\" links. <tipdesc>Link:</tipdesc> Add \"+\" and \"–\" links to modified versions of the current search that include or exclude their respective tags. <tipdesc>Toggle:</tipdesc> Add toggle links that modify the search box with their respective tags. Clicking a toggle link will switch between a tag being included (+), excluded (–), potentially included among other tags (~), and removed (»). Right clicking a toggle link will immediately remove its tag. If a tag already exists in the search box or gets entered/removed through alternative means, the toggle link will automatically update to reflect the tag's current status. <tiphead>Note</tiphead>The remove option is intended for users above the basic user level that want to remove the links. For users that can't normally see the links and do not wish to see them, this setting should be set to disabled.", {txtOptions:["Disabled:disabled", "Remove:remove", "Link:link", "Toggle:toggle"]}),
show_banned: newOption("checkbox", false, "Show Banned", "Display all banned posts in the search, pool, popular, favorites, comments, notes, and favorite group listings."),
show_deleted: newOption("checkbox", false, "Show Deleted", "Display all deleted posts in the search, pool, popular, favorites, notes, and favorite group listings. <tiphead>Note</tiphead>When using this option, your Danbooru account settings should have \"deleted post filter\" set to no and \"show deleted children\" set to yes in order to function properly and minimize connections to Danbooru."),
show_loli: newOption("checkbox", false, "Show Loli", "Display loli posts in the search, pool, popular, favorites, comments, notes, and favorite group listings."),
show_resized_notice: newOption("dropdown", "all", "Show Resized Notice", "Set which image type(s) the purple notice bar about image resizing is allowed to display on. <tiphead>Tip</tiphead>When a sample and original image are available for a post, a new option for swapping between the sample and original image becomes available in the sidebar options menu. Even if you disable the resized notice bar, you will always have access to its main function.", {txtOptions:["None (Disabled):none", "Original:original", "Sample:sample", "Original & Sample:all"]}),
show_shota: newOption("checkbox", false, "Show Shota", "Display shota posts in the search, pool, popular, favorites, comments, notes, and favorite group listings."),
show_toddlercon: newOption("checkbox", false, "Show Toddlercon", "Display toddlercon posts in the search, pool, popular, favorites, comments, notes, and favorite group listings."),
single_color_borders: newOption("checkbox", false, "Single Color Borders", "Only use one color for each thumbnail border."),
thumb_info: newOption("dropdown", "disabled", "Thumbnail Info", "Display the score (★), favorite count (♥), and rating (S, Q, or E) for a post with its thumbnail. <tipdesc>Below:</tipdesc> Display the extra information below thumbnails. <tipdesc>Hover:</tipdesc> Display the extra information upon hovering over a thumbnail's area. <tiphead>Note</tiphead>Extra information will not be added to the thumbnails in the comments listing since the score and rating are already visible there. Instead, the number of favorites will be added next to the existing score display.", {txtOptions:["Disabled:disabled", "Below:below", "Hover:hover"]}),
thumbnail_count: newOption("dropdown", 0, "Thumbnail Count", "Change the number of thumbnails that display in the search, favorites, and notes listings.", {txtOptions:["Disabled:0"], numRange:[1,200]}),
track_new: newOption("checkbox", false, "Track New Posts", "Add a menu option titled \"new\" to the posts section submenu (between \"listing\" and \"upload\") that links to a customized search focused on keeping track of new posts.<tiphead>Note</tiphead>While browsing the new posts, the current page of posts is also tracked. If the new post listing is left, clicking the \"new\" link later on will attempt to pull up the posts where browsing was left off at.<tiphead>Tip</tiphead>If you would like to bookmark the new post listing, drag and drop the link to your bookmarks or right click it and bookmark/copy the location from the context menu."),
status_borders: borderSet(["deleted", true, "#000000", "solid", "post-status-deleted"], ["flagged", true, "#FF0000", "solid", "post-status-flagged"], ["pending", true, "#0000FF", "solid", "post-status-pending"], ["child", true, "#CCCC00", "solid", "post-status-has-parent"], ["parent", true, "#00FF00", "solid", "post-status-has-children"]),
tag_borders: borderSet(["loli", true, "#FFC0CB", "solid"], ["shota", true, "#66CCFF", "solid"], ["toddlercon", true, "#9370DB", "solid"], ["status:banned", true, "#000000", "solid"]),
thumb_cache_limit: newOption("dropdown", 5000, "Thumbnail Info Cache Limit", "Limit the number of thumbnail information entries cached in the browser.<tiphead>Note</tiphead>No actual thumbnails are cached. Only filename information used to speed up the display of hidden thumbnails is stored. Every 1000 entries is approximately equal to 0.1 megabytes of space.", {txtOptions:["Disabled:0"], numList:[1000,2000,3000,4000,5000,6000,7000,8000,9000,10000]}),
collapse_sidebar_data: {post: {}, thumb: {}},
track_new_data: {viewed: 0, viewing: 1}
},
quick_search: "",
search_add: {
active_links: {},
links: {},
old: ""
},
sections: { // Setting sections and ordering.
blacklist_options: newSection("general", ["blacklist_session_toggle", "blacklist_post_display", "blacklist_thumb_mark", "blacklist_highlight_color", "blacklist_thumb_controls", "blacklist_smart_view", "blacklist_add_bars"], "Options"),
border_options: newSection("general", ["custom_tag_borders", "custom_status_borders", "single_color_borders", "border_width", "border_spacing"], "Options"),
browse: newSection("general", ["show_loli", "show_shota", "show_toddlercon", "show_banned", "show_deleted", "thumbnail_count", "thumb_info", "post_link_new_window"], "Post Browsing"),
control: newSection("general", ["load_sample_first", "alternate_image_swap", "image_swap_mode", "post_resize", "post_resize_mode", "post_drag_scroll", "autoscroll_post", "disable_embedded_notes"], "Post Control"),
endless: newSection("general", ["endless_default", "endless_session_toggle", "endless_separator", "endless_scroll_limit", "endless_remove_dup", "endless_pause_interval", "endless_fill", "endless_preload"], "Endless Pages"),
notices: newSection("general", ["show_resized_notice", "minimize_status_notices", "hide_sign_up_notice", "hide_upgrade_notice", "hide_tos_notice", "hide_comment_notice", "hide_tag_notice", "hide_upload_notice", "hide_pool_notice", "hide_ban_notice"], "Notices"),
sidebar: newSection("general", ["remove_tag_headers", "post_tag_scrollbars", "search_tag_scrollbars", "autohide_sidebar", "fixed_sidebar", "collapse_sidebar"], "Tag Sidebar"),
misc: newSection("general", ["direct_downloads", "track_new", "clean_links", "arrow_nav", "post_tag_titles", "search_add", "page_counter", "comment_score", "quick_search"], "Misc."),
misc_layout: newSection("general", ["fixed_paginator", "move_save_search"], "Misc."),
script_settings: newSection("general", ["bypass_api", "manage_cookies", "enable_status_message", "resize_link_style", "override_blacklist", "override_resize", "override_sample", "thumb_cache_limit"], "Script Settings"),
status_borders: newSection("border", "status_borders", "Custom Status Borders", "When using custom status borders, the borders can be edited here. For easy color selection, use one of the many free tools on the internet like <a target=\"_blank\" href=\"http://www.quackit.com/css/css_color_codes.cfm\">this one</a>."),
tag_borders: newSection("border", "tag_borders", "Custom Tag Borders", "When using custom tag borders, the borders can be edited here. For easy color selection, use one of the many free tools on the internet like <a target=\"_blank\" href=\"http://www.quackit.com/css/css_color_codes.cfm\">this one</a>.")
},
settings: {
changed: {}
},
timers: {},
user: {} // User settings.
};
localStorageCheck();
loadSettings(); // Load user settings.
// Provide a session ID in order to detect XML requests carrying over from other pages.
window.bbbSession = new Date().getTime();
// Location variables.
var gLoc = danbLoc(); // Current location
var gLocRegex = new RegExp("\\b" + gLoc + "\\b");
// Script variables.
// Global
var show_loli = bbb.user.show_loli;
var show_shota = bbb.user.show_shota;
var show_toddlercon = bbb.user.show_toddlercon;
var show_banned = bbb.user.show_banned;
var deleted_shown = (gLoc === "search" && /^(?:any|deleted)$/i.test(getTagVar("status"))); // Check whether deleted posts are shown by default.
var show_deleted = deleted_shown || bbb.user.show_deleted;
var direct_downloads = bbb.user.direct_downloads;
var post_link_new_window = bbb.user.post_link_new_window;
var blacklist_session_toggle = bbb.user.blacklist_session_toggle;
var blacklist_post_display = bbb.user.blacklist_post_display;
var blacklist_thumb_mark = bbb.user.blacklist_thumb_mark;
var blacklist_highlight_color = bbb.user.blacklist_highlight_color;
var blacklist_add_bars = bbb.user.blacklist_add_bars;
var blacklist_thumb_controls = bbb.user.blacklist_thumb_controls;
var blacklist_smart_view = bbb.user.blacklist_smart_view;
var custom_tag_borders = bbb.user.custom_tag_borders;
var custom_status_borders = bbb.user.custom_status_borders;
var single_color_borders = bbb.user.single_color_borders;
var border_spacing = bbb.user.border_spacing;
var border_width = bbb.user.border_width;
var clean_links = bbb.user.clean_links;
var comment_score = bbb.user.comment_score;
var thumb_info = bbb.user.thumb_info;
var autohide_sidebar = gLocRegex.test(bbb.user.autohide_sidebar);
var fixed_sidebar = gLocRegex.test(bbb.user.fixed_sidebar);
var fixed_paginator = bbb.user.fixed_paginator;
var collapse_sidebar = bbb.user.collapse_sidebar;
var move_save_search = bbb.user.move_save_search;
var page_counter = bbb.user.page_counter;
var quick_search = bbb.user.quick_search;
var bypass_api = bbb.user.bypass_api;
var manage_cookies = bbb.user.manage_cookies;
var enable_status_message = bbb.user.enable_status_message;
var resize_link_style = bbb.user.resize_link_style;
var override_blacklist = bbb.user.override_blacklist;
var override_resize = bbb.user.override_resize;
var override_sample = bbb.user.override_sample;
var track_new = bbb.user.track_new;
var show_resized_notice = bbb.user.show_resized_notice;
var hide_sign_up_notice = bbb.user.hide_sign_up_notice;
var hide_upgrade_notice = bbb.user.hide_upgrade_notice;
var minimize_status_notices = bbb.user.minimize_status_notices;
var hide_tos_notice = bbb.user.hide_tos_notice;
var hide_comment_notice = bbb.user.hide_comment_notice;
var hide_tag_notice = bbb.user.hide_tag_notice;
var hide_upload_notice = bbb.user.hide_upload_notice;
var hide_pool_notice = bbb.user.hide_pool_notice;
var hide_ban_notice = bbb.user.hide_ban_notice;
// Search
var arrow_nav = bbb.user.arrow_nav;
var search_add = bbb.user.search_add;
var search_tag_scrollbars = bbb.user.search_tag_scrollbars;
var thumbnail_count = bbb.user.thumbnail_count;
var thumbnail_count_default = 20; // Number of thumbnails BBB should expect Danbooru to return by default.
var thumb_cache_limit = bbb.user.thumb_cache_limit;
// Post
var alternate_image_swap = bbb.user.alternate_image_swap;
var post_resize = accountSettingCheck("post_resize");
var post_resize_mode = bbb.user.post_resize_mode;
var post_drag_scroll = bbb.user.post_drag_scroll;
var load_sample_first = accountSettingCheck("load_sample_first");
var remove_tag_headers = bbb.user.remove_tag_headers;
var post_tag_scrollbars = bbb.user.post_tag_scrollbars;
var post_tag_titles = bbb.user.post_tag_titles;
var autoscroll_post = bbb.user.autoscroll_post;
var image_swap_mode = bbb.user.image_swap_mode;
var disable_embedded_notes = bbb.user.disable_embedded_notes;
// Endless
var endless_default = bbb.user.endless_default;
var endless_fill = bbb.user.endless_fill;
var endless_pause_interval = bbb.user.endless_pause_interval;
var endless_preload = bbb.user.endless_preload;
var endless_remove_dup = bbb.user.endless_remove_dup;
var endless_scroll_limit = bbb.user.endless_scroll_limit;
var endless_separator = bbb.user.endless_separator;
var endless_session_toggle = bbb.user.endless_session_toggle;
// Stored data
var status_borders = bbb.user.status_borders;
var tag_borders = bbb.user.tag_borders;
var collapse_sidebar_data = bbb.user.collapse_sidebar_data;
var track_new_data = bbb.user.track_new_data;
var script_blacklisted_tags = bbb.user.script_blacklisted_tags;
// Other data
var bbbHiddenImg = "";
var bbbBlacklistImg = "";
var bbbBlacklistIcon = "";
/* "INIT" */
modifyDanbScript();
customCSS(); // Contains the portions related to notices.
thumbInfo();
removeTagHeaders();
searchAdd();
minimizeStatusNotices();
postTagTitles();
trackNew();
injectSettings();
modifyPage();
autohideSidebar();
moveSaveSearch();
pageCounter();
quickSearch();
postLinkNewWindow();
commentScoreInit();
cleanLinks();
postDDL();
arrowNav();
fixLimit();
bbbHotkeys();
endlessInit();
delayMe(formatThumbnails); // Delayed to allow Danbooru to run first.
delayMe(blacklistInit); // Delayed to allow Danbooru to run first.
delayMe(fixedSidebar); // Delayed to allow Danbooru layout to finalize.
delayMe(collapseSidebar); // Delayed to allow Danbooru layout to finalize.
delayMe(fixedPaginator); // Delayed to allow Danbooru layout to finalize.
/* Functions */
/* Functions for XML API info */
function searchJSON(mode, optArg) {
// Figure out the desired URL for a JSON API request, trigger any necessary xml flag, and update the status message.
var url = location.href.split("#", 1)[0];
var idCache, idList, idSearch, page; // If/else variables.
if (mode === "search" || mode === "notes" || mode === "favorites") {
url = (allowUserLimit() ? updateURLQuery(url, {limit: thumbnail_count}) : url);
bbb.flags.thumbs_xml = true;
if (mode === "search")
fetchJSON(url.replace(/\/?(?:posts)?\/?(?:\?|$)/, "/posts.json?"), "search");
else if (mode === "notes")
fetchJSON(url.replace(/\/notes\/?(?:\?|$)/, "/notes.json?"), "notes");
else if (mode === "favorites")
fetchJSON(url.replace(/\/favorites\/?(?:\?|$)/, "/favorites.json?"), "favorites");
bbbStatus("posts", "new");
}
else if (mode === "popular" || mode === "popular_view") {
bbb.flags.thumbs_xml = true;
fetchJSON(url.replace(/\/(popular_view|popular)\/?/, "/$1.json"), mode);
bbbStatus("posts", "new");
}
else if (mode === "pool" || mode === "favorite_group") {
idCache = getIdCache();
bbb.flags.thumbs_xml = true;
if (idCache)
searchJSON(mode + "_search", {post_ids: idCache});
else // Get a new cache.
fetchJSON(url.replace(/\/(pools|favorite_groups)\/(\d+)/, "/$1/$2.json"), mode + "_cache", mode + "_search");
bbbStatus("posts", "new");
}
else if (mode === "pool_search" || mode === "favorite_group_search") {
page = Number(getVar("page")) || 1;
idList = optArg.post_ids.split(" ");
idSearch = idList.slice((page - 1) * thumbnail_count_default, page * thumbnail_count_default);
fetchJSON("/posts.json?tags=status:any+id:" + idSearch.join(","), mode, idSearch);
}
else if (mode === "endless") {
bbb.flags.endless_xml = true;
if (gLoc === "pool" || gLoc === "favorite_group") {
idCache = getIdCache();
if (idCache)
searchJSON("endless_" + gLoc + "_search", {post_ids: idCache});
else // Get a new cache.
fetchJSON(url.replace(/\/(pools|favorite_groups)\/(\d+)/, "/$1/$2.json"), gLoc + "_cache", "endless_" + gLoc + "_search");
}
else {
url = endlessNexURL();
fetchJSON(url.replace(/(\?)|$/, ".json$1"), "endless");
}
bbbStatus("posts", "new");
}
else if (mode === "endless_pool_search" || mode === "endless_favorite_group_search") {
idList = optArg.post_ids.split(" ");
page = Number(getVar("page", endlessNexURL())); // If a pool gets over 1000 pages, I have no idea what happens for regular users. Biggest pool is currently around 400 pages so we won't worry about that for the time being.
idSearch = idList.slice((page - 1) * thumbnail_count_default, page * thumbnail_count_default);
fetchJSON("/posts.json?tags=status:any+id:" + idSearch.join(","), "endless", idSearch);
}
else if (mode === "comments") {
fetchJSON(url.replace(/\/comments\/?/, "/comments.json"), "comments");
bbbStatus("posts", "new");
}
else if (mode === "parent" || mode === "child") {
var parentUrl = "/posts.json?limit=200&tags=status:any+parent:" + optArg;
fetchJSON(parentUrl, mode, optArg);
bbbStatus("posts", "new");
}
else if (mode === "ugoira") {
fetchJSON(url.replace(/\/posts\/(\d+)/, "/posts/$1.json"), "ugoira");
bbbStatus("posts", "new");
}
}
function fetchJSON(url, mode, optArg, session, retries) {
// Retrieve JSON.
var xmlhttp = new XMLHttpRequest();
var xmlRetries = retries || 0;
var xmlSession = session || window.bbbSession;
if (xmlhttp !== null) {
xmlhttp.onreadystatechange = function() {
if (xmlSession !== window.bbbSession) // If we end up receiving an xml response from a different page, reject it.
xmlhttp.abort();
else if (xmlhttp.readyState === 4) { // 4 = "loaded"
if (xmlhttp.status === 200) { // 200 = "OK"
var xml = JSON.parse(xmlhttp.responseText);
// Update status message.
if (mode === "search" || mode === "popular" || mode === "popular_view" || mode === "notes" || mode === "favorites" || mode === "pool_search" || mode === "favorite_group_search") {
bbb.flags.thumbs_xml = false;
parseListing(xml, optArg);
}
else if (mode === "post")
parsePost(xml);
else if (mode === "pool_cache" || mode === "favorite_group_cache") {
var collId = /\/(?:pools|favorite_groups)\/(\d+)/.exec(location.href)[1];
sessionStorage.bbbSetItem("bbb_" + mode + "_" + collId, new Date().getTime() + " " + xml.post_ids);
searchJSON(optArg, xml);
}
else if (mode === "endless") {
bbb.flags.endless_xml = false;
endlessXMLJSONHandler(xml, optArg);
}
else if (mode === "comments")
parseComments(xml);
else if (mode === "parent" || mode === "child")
parseRelations(xml, mode, optArg);
else if (mode === "ugoira")
fixHiddenUgoira(xml);
if (mode !== "pool_cache" && mode !== "favorite_group_cache")
bbbStatus("posts", "done");
}
else {
if (xmlhttp.status === 403 || xmlhttp.status === 401) {
bbbNotice('Error retrieving post information. Access denied. You must be logged in to a Danbooru account to access the API for hidden image information and direct downloads. <br><span style="font-size: smaller;">(<span><a href="#" id="bbb-bypass-api-link">Do not warn me again and automatically bypass API features in the future.</a></span>)</span>', -1);
document.getElementById("bbb-bypass-api-link").addEventListener("click", function(event) {
if (event.button !== 0)
return;
updateSettings("bypass_api", true);
this.parentNode.innerHTML = "Settings updated. You may change this setting under the preferences tab in the settings panel.";
event.preventDefault();
}, false);
bbbStatus("posts", "error");
}
else if (xmlhttp.status === 421) {
bbbNotice("Error retrieving post information. Your Danbooru API access is currently throttled. Please try again later.", -1);
bbbStatus("posts", "error");
}
else if (xmlhttp.status !== 0) {
if (xmlRetries < 1) {
xmlRetries++;
fetchJSON(url, mode, optArg, xmlSession, xmlRetries);
}
else {
var linkId = uniqueIdNum(); // Create a unique ID.
var noticeMsg = bbbNotice('Error retrieving post information (JSON Code: ' + xmlhttp.status + ' ' + xmlhttp.statusText + '). (<a id="' + linkId + '" href="#">Retry</a>)', -1);
bbbStatus("posts", "error");
document.getElementById(linkId).addEventListener("click", function(event) {
if (event.button !== 0)
return;
closeBbbNoticeMsg(noticeMsg);
searchJSON(mode, optArg);
event.preventDefault();
}, false);
}
}
}
}
};
xmlhttp.open("GET", url, true);
xmlhttp.send(null);
}
}
function parseListing(xml, optArg) {
// Use JSON results for thumbnail listings.
var posts = xml;
var orderedIds = (gLoc === "pool" || gLoc === "favorite_group" ? optArg : undefined);
if (!posts[0])
return;
// Thumb preparation.
var newThumbs = createThumbListing(posts, orderedIds);
// Update the existing thumbnails with new ones.
updateThumbListing(newThumbs);
// Fix the paginator. The paginator isn't always in the new container, so run this on the whole page after the new container is inserted.
fixPaginator();
// Fix hidden thumbnails.
fixHiddenThumbs();
// Update the URL with the limit value.
fixURLLimit();
// Cache thumbnails to the history for random searches.
saveStateCache();
}
function parsePost(postInfo) {
// Take a post's info and alter its page.
var post = bbb.post.info = formatInfo(postInfo || scrapePost());
var imgContainer = document.getElementById("image-container");
if (!imgContainer) {
bbbNotice("Post content could not be located.", -1);
return;
}
if (!post || !post.file_url) {
bbbNotice("Due to a lack of provided information, this post cannot be viewed.", -1);
return;
}
// Stop if we're on Safebooru and the image isn't safe.
if (safebPostTest(post))
return;
// Enable the "Resize to window", "Toggle Notes", "Random Post", and "Find similar" options for logged out users.
createOptionsSection();
// Fix the direct post links in the information and options sections for hidden posts.
fixPostDownloadLinks();
// Replace the "resize to window" link with new resize links.
modifyResizeLink();
// Keep any original video from continuing to play/download after being removed.
var origVideo = imgContainer.getElementsByTagName("video")[0];
if (origVideo) {
origVideo.pause();
origVideo.src = "about:blank";
origVideo.load();
}
// Create content.
if (post.file_ext === "swf") // Create flash object.
imgContainer.innerHTML = '<div id="note-container"></div> <div id="note-preview"></div> <object height="' + post.image_height + '" width="' + post.image_width + '"> <params name="movie" value="' + post.file_url + '"> <embed allowscriptaccess="never" src="' + post.file_url + '" height="' + post.image_height + '" width="' + post.image_width + '"> </params> </object> <p><a href="' + post.file_url + '">Save this flash (right click and save)</a></p>';
else if (post.file_ext === "webm" || post.file_ext === "mp4") { // Create video
var playerLoop = (post.has_sound ? '' : ' loop="loop"'); // No looping for videos with sound.
imgContainer.innerHTML = '<div id="note-container"></div> <div id="note-preview"></div> <video id="image" autoplay="autoplay"' + playerLoop + ' controls="controls" src="' + post.file_url + '" height="' + post.image_height + '" width="' + post.image_width + '"></video> <p><a href="' + post.file_url + '">Save this video (right click and save)</a></p>';
}
else if (post.file_ext === "zip" && /(?:^|\s)ugoira(?:$|\s)/.test(post.tag_string)) { // Create ugoira
var useUgoiraOrig = getVar("original");
// Get rid of all the old events handlers.
if (Danbooru.Ugoira && Danbooru.Ugoira.player)
$(Danbooru.Ugoira.player).unbind();
if ((load_sample_first && useUgoiraOrig !== "1") || useUgoiraOrig === "0") { // Load sample webm version.
imgContainer.innerHTML = '<div id="note-container"></div> <div id="note-preview"></div> <video id="image" autoplay="autoplay" loop="loop" controls="controls" src="' + post.large_file_url + '" height="' + post.image_height + '" width="' + post.image_width + '" data-fav-count="' + post.fav_count + '" data-flags="' + post.flags + '" data-has-active-children="' + post.has_active_children + '" data-has-children="' + post.has_children + '" data-large-height="' + post.sample_height + '" data-large-width="' + post.sample_width + '" data-original-height="' + post.image_height + '" data-original-width="' + post.image_width + '" data-rating="' + post.rating + '" data-score="' + post.score + '" data-tags="' + post.tag_string + '" data-pools="' + post.pool_string + '" data-uploader="' + post.uploader_name + '"></video> <p><a href="' + post.large_file_url + '">Save this video (right click and save)</a> | <a href="' + updateURLQuery(location.href, {original: "1"}) + '">View original</a> | <a href="#" id="bbb-note-toggle">Toggle notes</a></p>';
// Prep the "toggle notes" link.
noteToggleLinkInit();
}
else { // Load original ugoira version.
imgContainer.innerHTML = '<div id="note-container"></div> <div id="note-preview"></div> <canvas data-ugoira-content-type="' + post.pixiv_ugoira_frame_data.content_type.replace(/"/g, """) + '" data-ugoira-frames="' + JSON.stringify(post.pixiv_ugoira_frame_data.data).replace(/"/g, """) + '" data-fav-count="' + post.fav_count + '" data-flags="' + post.flags + '" data-has-active-children="' + post.has_active_children + '" data-has-children="' + post.has_children + '" data-large-height="' + post.image_height + '" data-large-width="' + post.image_width + '" data-original-height="' + post.image_height + '" data-original-width="' + post.image_width + '" data-rating="' + post.rating + '" data-score="' + post.score + '" data-tags="' + post.tag_string + '" data-pools="' + post.pool_string + '" data-uploader="' + post.uploader_name + '" height="' + post.image_height + '" width="' + post.image_width + '" id="image"></canvas> <div id="ugoira-controls"> <div id="ugoira-control-panel" style="width: ' + post.image_width + 'px; min-width: 350px;"> <button id="ugoira-play" name="button" style="display: none;" type="submit">Play</button> <button id="ugoira-pause" name="button" type="submit">Pause</button> <div id="seek-slider" style="width: ' + (post.image_width - 81) + 'px; min-width: 269px;"></div> </div> <p id="save-video-link"><a href="' + post.large_file_url + '">Save as video (right click and save)</a> | <a href="' + updateURLQuery(location.href, {original: "0"}) + '">View sample</a> | <a href="#" id="bbb-note-toggle">Toggle notes</a></p> </div>';
// Make notes toggle when clicking the ugoira animation.
noteToggleInit();
// Prep the "toggle notes" link. The "toggle notes" link is added here just for consistency's sake.
noteToggleLinkInit();
if (post.pixiv_ugoira_frame_data.data) // Set up the post.
ugoiraInit();
else // Fix hidden posts.
searchJSON("ugoira");
}
}
else if (!post.image_height) // Create manual download.
imgContainer.innerHTML = '<div id="note-container"></div> <div id="note-preview"></div><p><a href="' + post.file_url + '">Save this file (right click and save)</a></p>';
else { // Create image
var newWidth, newHeight, newUrl; // If/else variables.
var imgDesc = (getMeta("og:title") || "").replace(" - Danbooru", "");
if (load_sample_first && post.has_large) {
newWidth = post.sample_width;
newHeight = post.sample_height;
newUrl = post.large_file_url;
}
else {
newWidth = post.image_width;
newHeight = post.image_height;
newUrl = post.file_url;
}
imgContainer.innerHTML = '<div id="note-container"></div> <div id="note-preview"></div> <img alt="' + post.tag_string + '" data-fav-count="' + post.fav_count + '" data-flags="' + post.flags + '" data-has-active-children="' + post.has_active_children + '" data-has-children="' + post.has_children + '" data-large-height="' + post.sample_height + '" data-large-width="' + post.sample_width + '" data-original-height="' + post.image_height + '" data-original-width="' + post.image_width + '" data-rating="' + post.rating + '" data-score="' + post.score + '" data-tags="' + post.tag_string + '" data-pools="' + post.pool_string + '" data-uploader="' + post.uploader_name + '" height="' + newHeight + '" width="' + newWidth + '" id="image" src="' + newUrl + '" /> <img src="about:blank" height="1" width="1" id="bbb-loader" style="position: absolute; right: 0px; top: 0px; display: none;"/> <p class="desc">' + imgDesc + '</p>';
bbb.el.bbbLoader = document.getElementById("bbb-loader");
// Create/replace the elements related to image swapping and set them up.
swapImageInit();
if (alternate_image_swap) // Make sample/original images swap when clicking the image.
alternateImageSwap();
else // Make notes toggle when clicking the image.
noteToggleInit();
}
// Enable drag scrolling.
dragScrollInit();
// Resize the content if desired.
if (post_resize)
resizePost(post_resize_mode);
// Enable translation mode.
translationModeInit();
// Disable embedded notes.
disableEmbeddedNotes();
// Load/reload notes.
Danbooru.Note.load_all("bbb");
// Auto position the content if desired.
autoscrollPost();
// Blacklist.
blacklistUpdate();
// Fix the parent/child notice(s).
checkRelations();
}
function parseComments(xml) {
// Fix missing comments by inserting them into their appropriate position.
var posts = xml;
var numPosts = posts.length;
var expectedPosts = numPosts;
var existingPosts = getPosts();
var eci = 0;
for (var i = 0; i < numPosts; i++) {
var post = formatInfo(posts[i]);
var existingPost = existingPosts[eci];
if (!existingPost || String(post.id) !== existingPost.getAttribute("data-id")) {
if (!/(?:^|\s)(?:loli|shota|toddlercon)(?:$|\s)/.test(post.tag_string) && !post.is_banned) // API post isn't hidden and doesn't exist on the page. Skip it and try to find where the page's info matches up.
continue;
else if ((!show_loli && /(?:^|\s)loli(?:$|\s)/.test(post.tag_string)) || (!show_shota && /(?:^|\s)shota(?:$|\s)/.test(post.tag_string)) || (!show_toddlercon && /(?:^|\s)toddlercon(?:$|\s)/.test(post.tag_string)) || (!show_banned && post.is_banned) || safebPostTest(post)) { // Skip hidden posts if the user has selected to do so.
expectedPosts--;
continue;
}
// Prepare the post information.
var tagLinks = post.tag_string.bbbSpacePad();
var generalTags = post.tag_string_general.split(" ");
var artistTags = post.tag_string_artist.split(" ");
var copyrightTags = post.tag_string_copyright.split(" ");
var characterTags = post.tag_string_character.split(" ");
var limit = (thumbnail_count ? "&limit=" + thumbnail_count : "");
var j, jl, tag; // Loop variables.
for (j = 0, jl = generalTags.length; j < jl; j++) {
tag = generalTags[j];
tagLinks = tagLinks.replace(tag.bbbSpacePad(), ' <span class="category-0"> <a href="/posts?tags=' + encodeURIComponent(tag) + limit + '">' + tag.replace(/_/g, " ") + '</a> </span> ');
}
for (j = 0, jl = artistTags.length; j < jl; j++) {
tag = artistTags[j];
tagLinks = tagLinks.replace(tag.bbbSpacePad(), ' <span class="category-1"> <a href="/posts?tags=' + encodeURIComponent(tag) + limit + '">' + tag.replace(/_/g, " ") + '</a> </span> ');
}
for (j = 0, jl = copyrightTags.length; j < jl; j++) {
tag = copyrightTags[j];
tagLinks = tagLinks.replace(tag.bbbSpacePad(), ' <span class="category-3"> <a href="/posts?tags=' + encodeURIComponent(tag) + limit + '">' + tag.replace(/_/g, " ") + '</a> </span> ');
}
for (j = 0, jl = characterTags.length; j < jl; j++) {
tag = characterTags[j];
tagLinks = tagLinks.replace(tag.bbbSpacePad(), ' <span class="category-4"> <a href="/posts?tags=' + encodeURIComponent(tag) + limit + '">' + tag.replace(/_/g, " ") + '</a> </span> ');
}
// Create the new post.
var childSpan = document.createElement("span");
childSpan.innerHTML = '<div id="post_' + post.id + '" class="post post-preview' + post.thumb_class + '" data-tags="' + post.tag_string + '" data-pools="' + post.pool_string + '" data-uploader="' + post.uploader_name + '" data-rating="' + post.rating + '" data-flags="' + post.flags + '" data-score="' + post.score + '" data-parent-id="' + post.parent_id + '" data-has-children="' + post.has_children + '" data-id="' + post.id + '" data-has-sound="' + post.has_sound + '" data-width="' + post.image_width + '" data-height="' + post.image_height + '" data-approver-id="' + post.approver_id + '" data-fav-count="' + post.fav_count + '" data-pixiv-id="' + post.pixiv_id + '" data-md5="' + post.md5 + '" data-file-ext="' + post.file_ext + '" data-file-url="' + post.file_url + '" data-large-file-url="' + post.large_file_url + '" data-preview-file-url="' + post.preview_file_url + '"> <div class="preview"> <a href="/posts/' + post.id + '"> <img alt="' + post.md5 + '" src="' + post.preview_file_url + '" /> </a> </div> <div class="comments-for-post" data-post-id="' + post.id + '"> <div class="header"> <div class="row"> <span class="info"> <strong>Date</strong> <time datetime="' + post.created_at + '" title="' + post.created_at.replace(/(.+)T(.+)-(.+)/, "$1 $2 -$3") + '">' + post.created_at.replace(/(.+)T(.+):\d+-.+/, "$1 $2") + '</time> </span> <span class="info"> <strong>User</strong> <a href="/users/' + post.uploader_id + '">' + post.uploader_name + '</a> </span> <span class="info"> <strong>Rating</strong> ' + post.rating + ' </span> <span class="info"> <strong>Score</strong> <span> <span id="score-for-post-' + post.id + '">' + post.score + '</span> </span> </span> </div> <div class="row list-of-tags"> <strong>Tags</strong>' + tagLinks + '</div> </div> </div> <div class="clearfix"></div> </div>';
// Prepare thumbnails.
prepThumbnails(childSpan);
if (!existingPost) // There isn't a next post so append the new post to the end before the paginator.
document.getElementById("a-index").insertBefore(childSpan.firstElementChild, getPaginator());
else // Insert new post before the post that should follow it.
existingPost.parentNode.insertBefore(childSpan.firstElementChild, existingPost);
// Get the comments and image info.
searchPages("post_comments", post.id);
}
eci++;
}
// If we don't have the expected number of posts, the API info and page are too out of sync. (Message disabled to work around deleted comments until an accurate method is worked out.)
// if (existingPosts.length !== expectedPosts)
// bbbNotice("Loading of hidden post(s) failed. Please refresh.", -1);
}
function parseRelations(xml, mode, parentId) {
// Create a new parent/child notice.
var posts = xml;
var activePost = bbb.post.info;
var numPosts = posts.length;
var relationCookie = getCookie()["show-relationship-previews"];
var showPreview = (relationCookie === undefined || relationCookie === "1" ? true : false);
var childSpan = document.createElement("span");
var query = "?tags=parent:" + parentId + (show_deleted ? "+status:any" : "") + (thumbnail_count ? "&limit=" + thumbnail_count : "");
var thumbs = "";
var forceShowDeleted = activePost.is_deleted; // If the parent is deleted or the active post is deleted, all deleted posts are shown.
var parentDeleted = false;
var isSafebooru = (location.host.indexOf("safebooru") > -1 ? true : false);
var target, previewLinkId, previewLinkTxt, previewId, classes, msg, displayStyle; // If/else variables.
var i, post; // Loop variables.
// Figure out if the parent is deleted.
for (i = 0; i < numPosts; i++) {
post = posts[i];
if (post.id === parentId) {
parentDeleted = post.is_deleted;
forceShowDeleted = forceShowDeleted || parentDeleted;
}
}
// Set up the notice variables.
if (showPreview) {
previewLinkTxt = "« hide";
displayStyle = "block";
}
else {
previewLinkTxt = "show »";
displayStyle = "none";
}
if (mode === "child") {
target = document.getElementsByClassName("notice-child")[0];
previewLinkId = "has-parent-relationship-preview-link";