-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathinject.js
269 lines (248 loc) · 8 KB
/
inject.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
/**
* All connectors are defined here, instead of manifest.
*
* Matching connector is injected to the page after document_end event.
*
* Do not include jQuery - it is included by default.
*
*
* Supported fields:
*
* label
* - label to be shown in options to enable/disable the connector
* - be careful with renaming, as connector disable state depends on the label
*
* matches
* - array of positive matches in format described in Chrome Ext. Dev. guide
* - connectors are processed in order and the first match is used; you can use
* this behaviour to emulate exclude matches
*
* js
* - array of paths of files to be executed
* - all executions happen on or after 'document_end'
*
* allFrames (optional)
* - boolean value representing InjectDetails.allFrames
* - FALSE by default
*
*/
var connectors = [
/*{
label: "myspace",
description : "MySpace",
matches: ["*://myspace.com/*"],
js: ["connectors/myspace.js"]
},*/
{
label: "pandora",
description: "Pandora Internet Radio",
matches: ["*://www.pandora.com/*"],
js: ["connectors/pandora.js"]
},
/*{
label: "deezer",
matches: ["*://www.deezer.com/*"],
description : "Deezer",
js: ["connectors/deezer.js"]
},*/
{
label: "google",
matches: ["*://play.google.com/music/*"],
description: "Google Play Music",
js: ["connectors/googlemusic.js"]
},
/*{
label: "myspace",
matches: ["*://myspace.com/*"],
description : "MySpace",
js: ["connectors/myspace.js"]
},*/
/**
* There's a bug with Soundcloud that causes the injected script
* to be injected more than once resulting in duplicate plays, and actually
* everytime the page is open.
*/
{
label: "soundcloud",
description: "Soundcloud",
matches: ["*://soundcloud.com/*"],
js: ["connectors/soundcloud.js"]
},
/*{
label: "vk",
description : "VK",
matches: ["*://vk.com/*"],
js: ["connectors/vk.js"]
},*/
/*{
label: "iheart",
description : "iHeart Radio",
matches: ["*://*.iheart.com/*"],
js: ["connectors/iheart.js"]
},*/
/*{
label: "Gaana.com",
matches: ["*://gaana.com/*"],
js: ["connectors/gaana.js"]
},*/
{
label: "spotify",
description: "Spotify Web Player",
matches: ["https://play.spotify.com/*"],
js: ["connectors/spotify.js"]
},
{
label: "grooveshark",
description: "Gooveshark",
matches: ["*://grooveshark.com/*"],
js: ["connectors/grooveshark.js"]
},
{
label: "tracks",
description: "8-Tracks",
matches: ["*://8tracks.com/*"],
js: ["connectors/8tracks.js"]
},
{
label: "songza",
description: "Songza",
matches: ["*://songza.com/*"],
js: ["connectors/songza.js"]
},
{
label: "rdio",
description: "Rdio",
matches: ["*://www.rdio.com/*"],
js: ["connectors/rdio.js"]
},
/*{
label: "gaana",
description : "Gaana",
matches: ["*://www.gaana.com/*"],
js: ["connectors/gaana.js"]
},*/
{
label: "youtube",
description: "YouTube",
matches: ["*://www.youtube.com/*"],
js: ["connectors/youtube.js"]
},
{
label: "di",
description: "Digitally Imported",
matches: ["*://www.di.fm/*"],
js: ["connectors/di.js"]
}
];
/**
* Creates regex from single match pattern
*
* @author lacivert
* @param {String} input
* @returns RegExp
*/
function createPattern(input) {
if (typeof input !== 'string') return null;
var match_pattern = '^'
, regEscape = function (s) {
return s.replace(/[[^$.|?*+(){}\\]/g, '\\$&');
}
, result = /^(\*|https?|file|ftp|chrome-extension):\/\//.exec(input);
// Parse scheme
if (!result) return null;
input = input.substr(result[0].length);
match_pattern += result[1] === '*' ? 'https?://' : result[1] + '://';
// Parse host if scheme is not `file`
if (result[1] !== 'file') {
if (!(result = /^(?:\*|(\*\.)?([^\/*]+))/.exec(input))) return null;
input = input.substr(result[0].length);
if (result[0] === '*') { // host is '*'
match_pattern += '[^/]+';
} else {
if (result[1]) { // Subdomain wildcard exists
match_pattern += '(?:[^/]+\.)?';
}
// Append host (escape special regex characters)
match_pattern += regEscape(result[2]);// + '/';
}
}
// Add remainder (path)
match_pattern += input.split('*').map(regEscape).join('.*');
match_pattern += '$';
return new RegExp(match_pattern);
}
function isConnectorEnabled(player) {
var disabledArray = JSON.parse(localStorage["disabledConnectors"]);
var i = disabledArray.indexOf(player)
return (i == -1);
}
var injectScriptIfComplete = function (tab) {
var tabId = tab.id;
// wait for the Loaded event
if (tab.status !== 'complete')
return;
// run first available connector
var anyMatch = !connectors.every(function (connector) {
var matchOk = false;
connector.matches.forEach(function (match) {
matchOk = matchOk || createPattern(match).test(tab.url);
});
if (matchOk === true) {
console.log('connector ' + connector.label + ' matched for ' + tab.url);
//setActionIcon(ACTION_SITE_RECOGNIZED, tabId);
if (!isConnectorEnabled(connector.label)) {
//setActionIcon(ACTION_SITE_DISABLED, tabId);
return false; // break forEach
}
// Ping the content page to see if the script is already in place.
// In the future, connectors will have unified interface, so they will all support
// the 'ping' request. Right now only YouTube supports this, because it
// is the only site that uses ajax navigation via History API (which is quite hard to catch).
// Other connectors will work as usual.
//
// Sadly there is no way to silently check if the script has been already injected
// so we will see an error in the background console on load of every supported page
chrome.tabs.sendMessage(tabId, {type: 'ping'}, function (response) {
// if the message was sent to a non existing script or the script
// does not implement the 'ping' message, we get response==undefined;
if (!response) {
console.log('-- loaded for the first time, injecting the scripts');
// inject all scripts and jQuery, use slice to avoid mutating
var scripts = connector.js.slice(0);
scripts.unshift(JQUERY_PATH);
scripts.forEach(function (jsFile) {
var injectDetails = {
file: jsFile,
allFrames: connector.allFrames ? connector.allFrames : false
};
chrome.tabs.executeScript(tabId, injectDetails);
});
}
else {
console.log('-- subsequent ajax navigation, the scripts are already injected');
}
});
}
return !matchOk;
});
// hide page action if there is no match
if (!anyMatch) {
if (chrome.pageAction)
chrome.pageAction.hide(tabId);
}
};
/**
* Initially inject scripts into all open tabs
*/
chrome.tabs.getAllInWindow(null, function(tabs) {
for (var i = 0; i < tabs.length; i++) {
var tab = tabs[i];
injectScriptIfComplete(tab);
}
});
/**
* Injects connectors to tabs upon page loading
*/
chrome.tabs.onUpdated.addListener(function(tabId, changeInfo, tab) {
injectScriptIfComplete(tab);
});