This repository has been archived by the owner on Nov 8, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 4
/
nexdirect.user.js
212 lines (179 loc) · 20.5 KB
/
nexdirect.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
// ==UserScript==
// @name NexDirect v2
// @namespace http://nexerq.cf/
// @homepage https://github.com/NexerqDev/NexDirect
// @version 0.3.0
// @icon https://raw.githubusercontent.com/NexerqDev/NexDirect/master/Designs/logo.png
// @description Adds download button to page to use NexDirect & replaces the heart button on the listings -- You must visit the Settings panel (the logo in the bottom right) and register the URI scheme before you are able to use this script.
// @author Nicholas Tay (Nexerq / @n2468txd) <nexerq@gmail.com>
// @license https://github.com/NexerqDev/NexDirect/blob/master/LICENSE
// @match http*://osu.ppy.sh/*
// @grant none
// @updateURL https://raw.githubusercontent.com/NexerqDev/NexDirect/master/nexdirect.user.js
// ==/UserScript==
(function() {
'use strict';
var nexDirectImage = "";
var miniIcon = "";
function welcome() {
console.log("%cWelcome to NexDirect.", "font-size: 35px; line-height: 38px; font-weight: bold; color: white; text-shadow: 0 0 5px green;");
}
function log(t) {
console.log("%c[NexDirect] %c" + t, "font-weight: bold;", "");
}
function injectListingPage() {
// Replace the <3 icons with nexdirect
log("Injecting download links into beatmap listing page...");
var $heartIcons = $("div.beatmapListing .beatmap .bmlist-options i.icon-heart");
if (!$heartIcons || ($heartIcons.length < 1))
return log("Could not find any heart buttons to replace."); // rip
log("Found " + $heartIcons.length + " heart icons. Replacing...");
$heartIcons.each(function() {
var $this = $(this);
var $closestBeatmap = $this.closest(".beatmap");
if (!$closestBeatmap || $closestBeatmap.length < 1)
return log("Could not find closest beatmap for element");
var beatmapSetId = $closestBeatmap.attr("id");
if (!beatmapSetId)
return log("Could not find beatmap set ID for element");
$this.parent().attr("href", "nexdirect://" + beatmapSetId);
$this.replaceWith("<div class=\"icon-nexdirect\"></div>");
});
log("Injected download buttons.");
$('head').append("<style type=\"text/css\"> .icon-nexdirect { cursor: default; background-image: url(" + miniIcon + "); opacity: 0.5; width: 20px; height: 20px; display: block !important; margin-left: 2px; } .icon-nexdirect:hover { opacity: 1 } </style>");
log("Injected additional CSS.");
}
function injectDownloadPage() {
log("Injecting download button into beatmap download page...");
// Get the beatmap *SET* id
var $imgSrc = $("img.bmt").attr("src"); // jack from image preview src elem
if (!$imgSrc)
return log("Could not find image preview src for ID."); // rip
log("Found image preview src element.");
var beatmapSetId = $imgSrc.match(/\/(\d+)l\.jpg/);
if (!beatmapSetId || !beatmapSetId[1])
return log("Could not identify the beatmap set ID."); // rip
beatmapSetId = beatmapSetId[1]; // its match 1
var $downloadButton = $(".beatmapDownloadButton");
if (!$downloadButton || $downloadButton.length < 1) {
log("Could not find the beatmap download button. Probably not logged in, going to try use description box.");
$downloadButton = $(".posttext");
if (!$downloadButton || $downloadButton.length < 1) // it really shouldnt get here
return log("Still cannot find element to inject.");
}
if ($downloadButton.length > 1) {
// rotate 180 for lined up, usually dont because it stands out - https://stackoverflow.com/questions/14233341/how-can-i-rotate-an-html-div-90-degrees
log("Found a no video/osu!supporter button, injecting after that one and rotating for conformity.");
$downloadButton = $($downloadButton[0]);
$('head').append("<style type=\"text/css\"> .nexdirectDownload { -webkit-transform: rotate(180deg); -moz-transform: rotate(180deg); -o-transform: rotate(180deg); -ms-transform: rotate(180deg); transform: rotate(180deg); } </style>");
}
$downloadButton.before("<div class=\"nexdirectDownload beatmapDownloadButton\"><a href=\"nexdirect://" + beatmapSetId + "\"><img src=\"" + nexDirectImage + "\" alt=\"Download with NexDirect\"/></a></div>");
log("Injected download button.");
}
function injectNewListingPage(firstLoad) {
log("[new] Finding new download button(s) to load into beatmap listing page...");
var $dlIconBoxes = $("div.beatmapset-panel__icons-box:not([nexdirect-loaded])");
if (!$dlIconBoxes || ($dlIconBoxes.length < 1))
return log("Could not find any new download icons to add."); // rip
log("Found " + $dlIconBoxes.length + " new download icons. Injecting NexDirect icons...");
$dlIconBoxes.each(function() {
var $this = $(this);
var $idElem = $this.parents("div.beatmapset-panel").children("a.js-audio--play");
if (!$idElem || ($idElem.length < 1))
return log("Could not find ID elem."); // rip
var beatmapSetId = $idElem.attr("data-audio-url").match(/\/(\d+)\.mp3$/);
if (!beatmapSetId || !beatmapSetId[1])
return log("Could not identify the beatmap set ID."); // rip
beatmapSetId = beatmapSetId[1]; // its match 1
$this.attr("nexdirect-loaded", "true"); // give it an attr to mark beatmap set as loaded
$this.prepend("<a href=\"nexdirect://" + beatmapSetId + "\" class=\"beatmapset-panel__icon\"><img src=\"" + miniIcon + "\" alt=\"Download with NexDirect\" style=\"filter: brightness(0.35); -webkit-filter: brightness(0.35); opacity: 0.9; margin-bottom: 5px;\"/></a>");
});
log("Injected download icons to beatmap listing page.");
}
function injectNewDownloadPage() {
log("[new] Injecting download button into beatmap download page...");
// Get set id
var $audioElem = $(".js-audio--play");
if (!$audioElem || ($audioElem.length < 1))
return log("Could not find link element."); // rip
log("Found link element.");
var previewLinkMatch = $audioElem.attr("data-audio-url").match(/\/preview\/(\d+)\.mp3/);
if (!previewLinkMatch || !previewLinkMatch[1])
return log("Could not retrieve beatmap set ID."); // rip
var beatmapSetId = previewLinkMatch[1]; // set the group
log("Found beatmap set ID.");
var nexDirectInject = `
<a href="${"nexdirect://" + beatmapSetId}" data-turbolinks="false" class="btn-osu-big btn-osu-big--beatmapset-header" style="filter: hue-rotate(284deg); -webkit-filter: hue-rotate(284deg);">
<div class="btn-osu-big__content">
<div class="btn-osu-big__left">
<span class="btn-osu-big__text-top">NexDirect</span>
</div>
<div class="btn-osu-big__icon">
<img src="${miniIcon}" alt="NexDirect icon" style="filter: brightness(0) invert(1); -webkit-filter: brightness(0) invert(1);">
</div>
</div>
</a>
`;
var $buttons = $(".beatmapset-header__buttons");
if (!$buttons || ($buttons.length < 1))
return log("Couldn't find buttons elem to inject.");
var $buttonAs = $buttons.find("a");
log("Injecting download button.");
if (!$buttonAs || ($buttonAs.length < 1)) {
// nicer to inject here if no buttons
log("No buttons, injecting to mapper portion instead.");
// needs wrapping when injecting in this spot
$(".beatmapset-mapping__content").after(`<div class="nexdirect--dl-wrapper" style="margin-left: 25px;">` + nexDirectInject + `</div>`);
} else {
$buttons.append(nexDirectInject);
}
var $osuDirectButton = $buttons.find("a:contains('osu!direct')");
if ($osuDirectButton.legnth > 0 && $osuDirectButton.attr("href").indexOf("osu://") < 0) { // no supporter, just get rid of that obsolete button
log("Found osu!supporter button, but user has no osu!supporter, deleting the obsolete element...");
$osuDirectButton.remove();
}
log("[new] Injected download button.");
}
function detectInjection(firstLoad) {
var newSite = typeof osu !== "undefined";
var path = window.location.pathname;
if (newSite) {
if (/\/beatmapsets\/(\d+)/.test(path)) {
if (!firstLoad)
log("----- PAGE CHANGE - beatmap page -----");
injectNewDownloadPage();
}
else if (path.indexOf("/beatmapsets") === 0) {
if (!firstLoad)
log("----- PAGE CHANGE - beatmap sets -----");
injectNewListingPage(true);
}
if (firstLoad) { // new site uses turbolinks, so use this to check page changes
log("[new] Attached turbolink cross-page handlers.");
var $document = $(document);
$document.on("turbolinks:load", function() { setTimeout(function() { detectInjection(); }, 650); }); // delay for page load change
$document.on("beatmap:search:done", function() { log("--- BEATMAP SEARCHED ---"); injectNewListingPage(); }); // beatmap search page load complete bind
$document.on("beatmap:load_more", function() {
if (NexDirect.loading) // need a loading var as for some reason the event gets fired 3 times each scroll as well
return;
NexDirect.loading = true;
log("--- VALID MORE BEATMAP LOADS REQUESTED ---");
setTimeout(function() {
injectNewListingPage();
setTimeout(function() { NexDirect.loading = false; }, 250); // allow loading again
}, 650); // delay to wait for load more to load and then inject again
});
}
} else {
if (path.indexOf("/p/beatmaplist") === 0)
injectListingPage();
else
injectDownloadPage();
}
}
window.NexDirect = {};
welcome();
setTimeout(function() {
detectInjection(true);
}, 650);
})();