ajaxlibs 0.1.11 → 0.1.12
Sign up to get free protection for your applications and to get access to all the features.
- data/README.rdoc +9 -3
- data/VERSION +1 -1
- data/lib/ajaxlibs.rb +2 -0
- data/lib/ajaxlibs/libraries/chrome_frame.rb +13 -0
- data/lib/ajaxlibs/libraries/webfont.rb +7 -0
- data/public/chrome-frame/1.0.0/CFInstall.js +224 -0
- data/public/chrome-frame/1.0.0/CFInstall.js.1 +224 -0
- data/public/chrome-frame/1.0.1/CFInstall.js +348 -0
- data/public/chrome-frame/1.0.1/CFInstall.min.js +11 -0
- data/public/chrome-frame/1.0.2/CFInstall.js +356 -0
- data/public/chrome-frame/1.0.2/CFInstall.min.js +5 -0
- data/public/webfont/1/webfont.js +35 -0
- data/public/webfont/1/webfont_debug.js +1092 -0
- data/spec/library_spec.rb +34 -34
- metadata +14 -4
@@ -0,0 +1,348 @@
|
|
1
|
+
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
|
2
|
+
// Use of this source code is governed by a BSD-style license that can be
|
3
|
+
// found in the LICENSE file.
|
4
|
+
|
5
|
+
/**
|
6
|
+
* @fileoverview CFInstall.js provides a set of utilities for managing
|
7
|
+
* the Chrome Frame detection and installation process.
|
8
|
+
* @author slightlyoff@google.com (Alex Russell)
|
9
|
+
*/
|
10
|
+
|
11
|
+
(function(scope) {
|
12
|
+
// bail if we'd be over-writing an existing CFInstall object
|
13
|
+
if (scope['CFInstall']) {
|
14
|
+
return;
|
15
|
+
}
|
16
|
+
|
17
|
+
/**
|
18
|
+
* returns an item based on DOM ID. Optionally a document may be provided to
|
19
|
+
* specify the scope to search in. If a node is passed, it's returned as-is.
|
20
|
+
* @param {string|Node} id The ID of the node to be located or a node
|
21
|
+
* @param {Node} doc Optional A document to search for id.
|
22
|
+
* @return {Node}
|
23
|
+
*/
|
24
|
+
var byId = function(id, doc) {
|
25
|
+
return (typeof id == 'string') ? (doc || document).getElementById(id) : id;
|
26
|
+
};
|
27
|
+
|
28
|
+
/////////////////////////////////////////////////////////////////////////////
|
29
|
+
// Plugin Detection
|
30
|
+
/////////////////////////////////////////////////////////////////////////////
|
31
|
+
|
32
|
+
/**
|
33
|
+
* Checks to find out if ChromeFrame is available as a plugin
|
34
|
+
* @return {Boolean}
|
35
|
+
*/
|
36
|
+
var isAvailable = function() {
|
37
|
+
// For testing purposes.
|
38
|
+
if (scope.CFInstall._force) {
|
39
|
+
return scope.CFInstall._forceValue;
|
40
|
+
}
|
41
|
+
|
42
|
+
// Look for CF in the User Agent before trying more expensive checks
|
43
|
+
var ua = navigator.userAgent.toLowerCase();
|
44
|
+
if (ua.indexOf("chromeframe") >= 0) {
|
45
|
+
return true;
|
46
|
+
}
|
47
|
+
|
48
|
+
if (typeof window['ActiveXObject'] != 'undefined') {
|
49
|
+
try {
|
50
|
+
var obj = new ActiveXObject('ChromeTab.ChromeFrame');
|
51
|
+
if (obj) {
|
52
|
+
return true;
|
53
|
+
}
|
54
|
+
} catch(e) {
|
55
|
+
// squelch
|
56
|
+
}
|
57
|
+
}
|
58
|
+
return false;
|
59
|
+
};
|
60
|
+
|
61
|
+
/**
|
62
|
+
* Creates a style sheet in the document containing the passed rules.
|
63
|
+
*/
|
64
|
+
var injectStyleSheet = function(rules) {
|
65
|
+
try {
|
66
|
+
var ss = document.createElement('style');
|
67
|
+
ss.setAttribute('type', 'text/css');
|
68
|
+
if (ss.styleSheet) {
|
69
|
+
ss.styleSheet.cssText = rules;
|
70
|
+
} else {
|
71
|
+
ss.appendChild(document.createTextNode(rules));
|
72
|
+
}
|
73
|
+
var h = document.getElementsByTagName('head')[0];
|
74
|
+
var firstChild = h.firstChild;
|
75
|
+
h.insertBefore(ss, firstChild);
|
76
|
+
} catch (e) {
|
77
|
+
// squelch
|
78
|
+
}
|
79
|
+
};
|
80
|
+
|
81
|
+
/** @type {boolean} */
|
82
|
+
var cfStyleTagInjected = false;
|
83
|
+
/** @type {boolean} */
|
84
|
+
var cfHiddenInjected = false;
|
85
|
+
|
86
|
+
/**
|
87
|
+
* Injects style rules into the document to handle formatting of Chrome Frame
|
88
|
+
* prompt. Multiple calls have no effect.
|
89
|
+
*/
|
90
|
+
var injectCFStyleTag = function() {
|
91
|
+
if (cfStyleTagInjected) {
|
92
|
+
// Once and only once
|
93
|
+
return;
|
94
|
+
}
|
95
|
+
var rules = '.chromeFrameInstallDefaultStyle {' +
|
96
|
+
'width: 800px;' +
|
97
|
+
'height: 600px;' +
|
98
|
+
'position: absolute;' +
|
99
|
+
'left: 50%;' +
|
100
|
+
'top: 50%;' +
|
101
|
+
'margin-left: -400px;' +
|
102
|
+
'margin-top: -300px;' +
|
103
|
+
'}' +
|
104
|
+
'.chromeFrameOverlayContent {' +
|
105
|
+
'position: absolute;' +
|
106
|
+
'margin-left: -400px;' +
|
107
|
+
'margin-top: -300px;' +
|
108
|
+
'left: 50%;' +
|
109
|
+
'top: 50%;' +
|
110
|
+
'border: 1px solid #93B4D9;' +
|
111
|
+
'background-color: white;' +
|
112
|
+
'}' +
|
113
|
+
'.chromeFrameOverlayContent iframe {' +
|
114
|
+
'width: 800px;' +
|
115
|
+
'height: 600px;' +
|
116
|
+
'border: none;' +
|
117
|
+
'}' +
|
118
|
+
'.chromeFrameOverlayCloseBar {' +
|
119
|
+
'height: 1em;' +
|
120
|
+
'text-align: right;' +
|
121
|
+
'background-color: #CADEF4;' +
|
122
|
+
'}' +
|
123
|
+
'.chromeFrameOverlayUnderlay {' +
|
124
|
+
'position: absolute;' +
|
125
|
+
'width: 100%;' +
|
126
|
+
'height: 100%;' +
|
127
|
+
'background-color: white;' +
|
128
|
+
'opacity: 0.5;' +
|
129
|
+
'-moz-opacity: 0.5;' +
|
130
|
+
'-webkit-opacity: 0.5;' +
|
131
|
+
'-ms-filter: ' +
|
132
|
+
'"progid:DXImageTransform.Microsoft.Alpha(Opacity=50)";' +
|
133
|
+
'filter: alpha(opacity=50);' +
|
134
|
+
'}';
|
135
|
+
injectStyleSheet(rules);
|
136
|
+
cfStyleTagInjected = true;
|
137
|
+
};
|
138
|
+
|
139
|
+
/**
|
140
|
+
* Injects style rules to hide the overlay version of the GCF prompt.
|
141
|
+
* Multiple calls have no effect.
|
142
|
+
*/
|
143
|
+
var closeOverlay = function() {
|
144
|
+
// IE has a limit to the # of <style> tags allowed, so we avoid
|
145
|
+
// tempting the fates.
|
146
|
+
if (cfHiddenInjected) {
|
147
|
+
return;
|
148
|
+
}
|
149
|
+
var rules = '.chromeFrameOverlayContent { display: none; }' +
|
150
|
+
'.chromeFrameOverlayUnderlay { display: none; }';
|
151
|
+
injectStyleSheet(rules);
|
152
|
+
// Hide the dialog for a year (or until cookies are deleted).
|
153
|
+
var age = 365 * 24 * 60 * 60 * 1000;
|
154
|
+
document.cookie = "disableGCFCheck=1;path=/;max-age="+age;
|
155
|
+
cfHiddenInjected = true;
|
156
|
+
};
|
157
|
+
|
158
|
+
/**
|
159
|
+
* Plucks properties from the passed arguments and sets them on the passed
|
160
|
+
* DOM node
|
161
|
+
* @param {Node} node The node to set properties on
|
162
|
+
* @param {Object} args A map of user-specified properties to set
|
163
|
+
*/
|
164
|
+
var setProperties = function(node, args) {
|
165
|
+
|
166
|
+
var srcNode = byId(args['node']);
|
167
|
+
|
168
|
+
node.id = args['id'] || (srcNode ? srcNode['id'] || getUid(srcNode) : '');
|
169
|
+
|
170
|
+
// TODO(slightlyoff): Opera compat? need to test there
|
171
|
+
var cssText = args['cssText'] || '';
|
172
|
+
node.style.cssText = ' ' + cssText;
|
173
|
+
|
174
|
+
var classText = args['className'] || '';
|
175
|
+
node.className = classText;
|
176
|
+
|
177
|
+
// default if the browser doesn't so we don't show sad-tab
|
178
|
+
var src = args['src'] || 'about:blank';
|
179
|
+
|
180
|
+
node.src = src;
|
181
|
+
|
182
|
+
if (srcNode) {
|
183
|
+
srcNode.parentNode.replaceChild(node, srcNode);
|
184
|
+
}
|
185
|
+
};
|
186
|
+
|
187
|
+
/**
|
188
|
+
* Creates an iframe.
|
189
|
+
* @param {Object} args A bag of configuration properties, including values
|
190
|
+
* like 'node', 'cssText', 'className', 'id', 'src', etc.
|
191
|
+
* @return {Node}
|
192
|
+
*/
|
193
|
+
var makeIframe = function(args) {
|
194
|
+
var el = document.createElement('iframe');
|
195
|
+
el.setAttribute('frameborder', '0');
|
196
|
+
el.setAttribute('border', '0');
|
197
|
+
setProperties(el, args);
|
198
|
+
return el;
|
199
|
+
};
|
200
|
+
|
201
|
+
/**
|
202
|
+
* Adds an unadorned iframe into the page, taking arguments to customize it.
|
203
|
+
* @param {Object} args A map of user-specified properties to set
|
204
|
+
*/
|
205
|
+
var makeInlinePrompt = function(args) {
|
206
|
+
args.className = 'chromeFrameInstallDefaultStyle ' +
|
207
|
+
(args.className || '');
|
208
|
+
var ifr = makeIframe(args);
|
209
|
+
// TODO(slightlyoff): handle placement more elegantly!
|
210
|
+
if (!ifr.parentNode) {
|
211
|
+
var firstChild = document.body.firstChild;
|
212
|
+
document.body.insertBefore(ifr, firstChild);
|
213
|
+
}
|
214
|
+
};
|
215
|
+
|
216
|
+
/**
|
217
|
+
* Adds a styled, closable iframe into the page with a background that
|
218
|
+
* emulates a modal dialog.
|
219
|
+
* @param {Object} args A map of user-specified properties to set
|
220
|
+
*/
|
221
|
+
var makeOverlayPrompt = function(args) {
|
222
|
+
if (byId('chromeFrameOverlayContent')) {
|
223
|
+
return; // Was previously created. Bail.
|
224
|
+
}
|
225
|
+
|
226
|
+
var n = document.createElement('span');
|
227
|
+
n.innerHTML = '<div class="chromeFrameOverlayUnderlay"></div>' +
|
228
|
+
'<table class="chromeFrameOverlayContent"' +
|
229
|
+
'id="chromeFrameOverlayContent"' +
|
230
|
+
'cellpadding="0" cellspacing="0">' +
|
231
|
+
'<tr class="chromeFrameOverlayCloseBar">' +
|
232
|
+
'<td>' +
|
233
|
+
// TODO(slightlyoff): i18n
|
234
|
+
'<button id="chromeFrameCloseButton">close</button>' +
|
235
|
+
'</td>' +
|
236
|
+
'</tr>' +
|
237
|
+
'<tr>' +
|
238
|
+
'<td id="chromeFrameIframeHolder"></td>' +
|
239
|
+
'</tr>' +
|
240
|
+
'</table>';
|
241
|
+
|
242
|
+
document.body.appendChild(n);
|
243
|
+
var ifr = makeIframe(args);
|
244
|
+
byId('chromeFrameIframeHolder').appendChild(ifr);
|
245
|
+
byId('chromeFrameCloseButton').onclick = closeOverlay;
|
246
|
+
};
|
247
|
+
|
248
|
+
var CFInstall = {};
|
249
|
+
|
250
|
+
/**
|
251
|
+
* Checks to see if Chrome Frame is available, if not, prompts the user to
|
252
|
+
* install. Once installation is begun, a background timer starts,
|
253
|
+
* checkinging for a successful install every 2 seconds. Upon detection of
|
254
|
+
* successful installation, the current page is reloaded, or if a
|
255
|
+
* 'destination' parameter is passed, the page navigates there instead.
|
256
|
+
* @param {Object} args A bag of configuration properties. Respected
|
257
|
+
* properties are: 'mode', 'url', 'destination', 'node', 'onmissing',
|
258
|
+
* 'preventPrompt', 'oninstall', 'preventInstallDetection', 'cssText', and
|
259
|
+
* 'className'.
|
260
|
+
* @public
|
261
|
+
*/
|
262
|
+
CFInstall.check = function(args) {
|
263
|
+
args = args || {};
|
264
|
+
|
265
|
+
// We currently only support CF in IE
|
266
|
+
// TODO(slightlyoff): Update this should we support other browsers!
|
267
|
+
var ua = navigator.userAgent;
|
268
|
+
var ieRe = /MSIE \S+; Windows NT/;
|
269
|
+
var bail = false;
|
270
|
+
if (ieRe.test(ua)) {
|
271
|
+
// We also only support Win2003/XPSP2 or better. See:
|
272
|
+
// http://msdn.microsoft.com/en-us/library/ms537503%28VS.85%29.aspx
|
273
|
+
if (parseFloat(ua.split(ieRe)[1]) < 6 &&
|
274
|
+
ua.indexOf('SV1') >= 0) {
|
275
|
+
bail = true;
|
276
|
+
}
|
277
|
+
} else {
|
278
|
+
bail = true;
|
279
|
+
}
|
280
|
+
if (bail) {
|
281
|
+
return;
|
282
|
+
}
|
283
|
+
|
284
|
+
// Inject the default styles
|
285
|
+
injectCFStyleTag();
|
286
|
+
|
287
|
+
if (document.cookie.indexOf("disableGCFCheck=1") >=0) {
|
288
|
+
// If we're supposed to hide the overlay prompt, add the rules to do it.
|
289
|
+
closeOverlay();
|
290
|
+
}
|
291
|
+
|
292
|
+
// When loaded in an alternate protocol (e.g., "file:"), still call out to
|
293
|
+
// the right location.
|
294
|
+
var currentProtocol = document.location.protocol;
|
295
|
+
var protocol = (currentProtocol == 'https:') ? 'https:' : 'http:';
|
296
|
+
// TODO(slightlyoff): Update this URL when a mini-installer page is
|
297
|
+
// available.
|
298
|
+
var installUrl = protocol + '//www.google.com/chromeframe';
|
299
|
+
if (!isAvailable()) {
|
300
|
+
if (args.onmissing) {
|
301
|
+
args.onmissing();
|
302
|
+
}
|
303
|
+
|
304
|
+
args.src = args.url || installUrl;
|
305
|
+
var mode = args.mode || 'inline';
|
306
|
+
var preventPrompt = args.preventPrompt || false;
|
307
|
+
|
308
|
+
if (!preventPrompt) {
|
309
|
+
if (mode == 'inline') {
|
310
|
+
makeInlinePrompt(args);
|
311
|
+
} else if (mode == 'overlay') {
|
312
|
+
makeOverlayPrompt(args);
|
313
|
+
} else {
|
314
|
+
window.open(args.src);
|
315
|
+
}
|
316
|
+
}
|
317
|
+
|
318
|
+
if (args.preventInstallDetection) {
|
319
|
+
return;
|
320
|
+
}
|
321
|
+
|
322
|
+
// Begin polling for install success.
|
323
|
+
var installTimer = setInterval(function() {
|
324
|
+
// every 2 seconds, look to see if CF is available, if so, proceed on
|
325
|
+
// to our destination
|
326
|
+
if (isAvailable()) {
|
327
|
+
if (args.oninstall) {
|
328
|
+
args.oninstall();
|
329
|
+
}
|
330
|
+
|
331
|
+
clearInterval(installTimer);
|
332
|
+
// TODO(slightlyoff): add a way to prevent navigation or make it
|
333
|
+
// contingent on oninstall?
|
334
|
+
window.location = args.destination || window.location;
|
335
|
+
}
|
336
|
+
}, 2000);
|
337
|
+
}
|
338
|
+
};
|
339
|
+
|
340
|
+
CFInstall._force = false;
|
341
|
+
CFInstall._forceValue = false;
|
342
|
+
CFInstall.isAvailable = isAvailable;
|
343
|
+
|
344
|
+
// expose CFInstall to the external scope. We've already checked to make
|
345
|
+
// sure we're not going to blow existing objects away.
|
346
|
+
scope.CFInstall = CFInstall;
|
347
|
+
|
348
|
+
})(this['ChromeFrameInstallScope'] || this);
|
@@ -0,0 +1,11 @@
|
|
1
|
+
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
|
2
|
+
// Use of this source code is governed by a BSD-style license that can be
|
3
|
+
// found in the LICENSE file.
|
4
|
+
|
5
|
+
(function(f){if(!f.CFInstall){var g=function(a,b){return typeof a=="string"?(b||document).getElementById(a):a},h=function(){if(f.CFInstall._force)return f.CFInstall._forceValue;var a=navigator.userAgent.toLowerCase();if(a.indexOf("chromeframe")>=0)return true;if(typeof window.ActiveXObject!="undefined")try{var b=new ActiveXObject("ChromeTab.ChromeFrame");if(b)return true}catch(c){}return false},j=function(a){try{var b=document.createElement("style");b.setAttribute("type","text/css");if(b.styleSheet)b.styleSheet.cssText=
|
6
|
+
a;else b.appendChild(document.createTextNode(a));var c=document.getElementsByTagName("head")[0],d=c.firstChild;c.insertBefore(b,d)}catch(i){}},k=false,l=false,o=function(){if(!k){var a='.chromeFrameInstallDefaultStyle {width: 800px;height: 600px;position: absolute;left: 50%;top: 50%;margin-left: -400px;margin-top: -300px;}.chromeFrameOverlayContent {position: absolute;margin-left: -400px;margin-top: -300px;left: 50%;top: 50%;border: 1px solid #93B4D9;background-color: white;}.chromeFrameOverlayContent iframe {width: 800px;height: 600px;border: none;}.chromeFrameOverlayCloseBar {height: 1em;text-align: right;background-color: #CADEF4;}.chromeFrameOverlayUnderlay {position: absolute;width: 100%;height: 100%;background-color: white;opacity: 0.5;-moz-opacity: 0.5;-webkit-opacity: 0.5;-ms-filter: "progid:DXImageTransform.Microsoft.Alpha(Opacity=50)";filter: alpha(opacity=50);}';
|
7
|
+
j(a);k=true}},m=function(){if(!l){var a=".chromeFrameOverlayContent { display: none; }.chromeFrameOverlayUnderlay { display: none; }";j(a);a=31536000000;document.cookie="disableGCFCheck=1;path=/;max-age="+a;l=true}},p=function(a,b){var c=g(b.node);a.id=b.id||(c?c.id||getUid(c):"");var d=b.cssText||"";a.style.cssText=" "+d;d=b.className||"";a.className=d;b=b.src||"about:blank";a.src=b;c&&c.parentNode.replaceChild(a,c)},n=function(a){var b=document.createElement("iframe");b.setAttribute("frameborder",
|
8
|
+
"0");b.setAttribute("border","0");p(b,a);return b},q=function(a){a.className="chromeFrameInstallDefaultStyle "+(a.className||"");a=n(a);if(!a.parentNode){var b=document.body.firstChild;document.body.insertBefore(a,b)}},r=function(a){if(!g("chromeFrameOverlayContent")){var b=document.createElement("span");b.innerHTML='<div class="chromeFrameOverlayUnderlay"></div><table class="chromeFrameOverlayContent"id="chromeFrameOverlayContent"cellpadding="0" cellspacing="0"><tr class="chromeFrameOverlayCloseBar"><td><button id="chromeFrameCloseButton">close</button></td></tr><tr><td id="chromeFrameIframeHolder"></td></tr></table>';
|
9
|
+
document.body.appendChild(b);a=n(a);g("chromeFrameIframeHolder").appendChild(a);g("chromeFrameCloseButton").onclick=m}},e={};e.check=function(a){a=a||{};var b=navigator.userAgent,c=/MSIE \S+; Windows NT/,d=false;if(c.test(b)){if(parseFloat(b.split(c)[1])<6&&b.indexOf("SV1")>=0)d=true}else d=true;if(!d){o();document.cookie.indexOf("disableGCFCheck=1")>=0&&m();b=document.location.protocol;b=b=="https:"?"https:":"http:";b=b+"//www.google.com/chromeframe";if(!h()){a.onmissing&&a.onmissing();a.src=a.url||
|
10
|
+
b;b=a.mode||"inline";c=a.preventPrompt||false;if(!c)if(b=="inline")q(a);else b=="overlay"?r(a):window.open(a.src);if(!a.preventInstallDetection)var i=setInterval(function(){if(h()){a.oninstall&&a.oninstall();clearInterval(i);window.location=a.destination||window.location}},2000)}}};e._force=false;e._forceValue=false;e.isAvailable=h;f.CFInstall=e}})(this.ChromeFrameInstallScope||this);
|
11
|
+
|
@@ -0,0 +1,356 @@
|
|
1
|
+
// Copyright (c) 2009 The Chromium Authors. All rights reserved.
|
2
|
+
// Use of this source code is governed by a BSD-style license that can be
|
3
|
+
// found in the LICENSE file.
|
4
|
+
|
5
|
+
/**
|
6
|
+
* @fileoverview CFInstall.js provides a set of utilities for managing
|
7
|
+
* the Chrome Frame detection and installation process.
|
8
|
+
* @author slightlyoff@google.com (Alex Russell)
|
9
|
+
*/
|
10
|
+
|
11
|
+
(function(scope) {
|
12
|
+
// bail if we'd be over-writing an existing CFInstall object
|
13
|
+
if (scope['CFInstall']) {
|
14
|
+
return;
|
15
|
+
}
|
16
|
+
|
17
|
+
/**
|
18
|
+
* returns an item based on DOM ID. Optionally a document may be provided to
|
19
|
+
* specify the scope to search in. If a node is passed, it's returned as-is.
|
20
|
+
* @param {string|Node} id The ID of the node to be located or a node
|
21
|
+
* @param {Node} doc Optional A document to search for id.
|
22
|
+
* @return {Node}
|
23
|
+
*/
|
24
|
+
var byId = function(id, doc) {
|
25
|
+
return (typeof id == 'string') ? (doc || document).getElementById(id) : id;
|
26
|
+
};
|
27
|
+
|
28
|
+
/////////////////////////////////////////////////////////////////////////////
|
29
|
+
// Plugin Detection
|
30
|
+
/////////////////////////////////////////////////////////////////////////////
|
31
|
+
|
32
|
+
/**
|
33
|
+
* Checks to find out if ChromeFrame is available as a plugin
|
34
|
+
* @return {Boolean}
|
35
|
+
*/
|
36
|
+
var isAvailable = function() {
|
37
|
+
// For testing purposes.
|
38
|
+
if (scope.CFInstall._force) {
|
39
|
+
return scope.CFInstall._forceValue;
|
40
|
+
}
|
41
|
+
|
42
|
+
// Look for CF in the User Agent before trying more expensive checks
|
43
|
+
var ua = navigator.userAgent.toLowerCase();
|
44
|
+
if (ua.indexOf("chromeframe") >= 0) {
|
45
|
+
return true;
|
46
|
+
}
|
47
|
+
|
48
|
+
if (typeof window['ActiveXObject'] != 'undefined') {
|
49
|
+
try {
|
50
|
+
var obj = new ActiveXObject('ChromeTab.ChromeFrame');
|
51
|
+
if (obj) {
|
52
|
+
return true;
|
53
|
+
}
|
54
|
+
} catch(e) {
|
55
|
+
// squelch
|
56
|
+
}
|
57
|
+
}
|
58
|
+
return false;
|
59
|
+
};
|
60
|
+
|
61
|
+
/**
|
62
|
+
* Creates a style sheet in the document containing the passed rules.
|
63
|
+
*/
|
64
|
+
var injectStyleSheet = function(rules) {
|
65
|
+
try {
|
66
|
+
var ss = document.createElement('style');
|
67
|
+
ss.setAttribute('type', 'text/css');
|
68
|
+
if (ss.styleSheet) {
|
69
|
+
ss.styleSheet.cssText = rules;
|
70
|
+
} else {
|
71
|
+
ss.appendChild(document.createTextNode(rules));
|
72
|
+
}
|
73
|
+
var h = document.getElementsByTagName('head')[0];
|
74
|
+
var firstChild = h.firstChild;
|
75
|
+
h.insertBefore(ss, firstChild);
|
76
|
+
} catch (e) {
|
77
|
+
// squelch
|
78
|
+
}
|
79
|
+
};
|
80
|
+
|
81
|
+
/** @type {boolean} */
|
82
|
+
var cfStyleTagInjected = false;
|
83
|
+
/** @type {boolean} */
|
84
|
+
var cfHiddenInjected = false;
|
85
|
+
|
86
|
+
/**
|
87
|
+
* Injects style rules into the document to handle formatting of Chrome Frame
|
88
|
+
* prompt. Multiple calls have no effect.
|
89
|
+
*/
|
90
|
+
var injectCFStyleTag = function() {
|
91
|
+
if (cfStyleTagInjected) {
|
92
|
+
// Once and only once
|
93
|
+
return;
|
94
|
+
}
|
95
|
+
var rules = '.chromeFrameInstallDefaultStyle {' +
|
96
|
+
'width: 800px;' +
|
97
|
+
'height: 600px;' +
|
98
|
+
'position: absolute;' +
|
99
|
+
'left: 50%;' +
|
100
|
+
'top: 50%;' +
|
101
|
+
'margin-left: -400px;' +
|
102
|
+
'margin-top: -300px;' +
|
103
|
+
'}' +
|
104
|
+
'.chromeFrameOverlayContent {' +
|
105
|
+
'position: absolute;' +
|
106
|
+
'margin-left: -400px;' +
|
107
|
+
'margin-top: -300px;' +
|
108
|
+
'left: 50%;' +
|
109
|
+
'top: 50%;' +
|
110
|
+
'border: 1px solid #93B4D9;' +
|
111
|
+
'background-color: white;' +
|
112
|
+
'z-index: 2001;' +
|
113
|
+
'}' +
|
114
|
+
'.chromeFrameOverlayContent iframe {' +
|
115
|
+
'width: 800px;' +
|
116
|
+
'height: 600px;' +
|
117
|
+
'border: none;' +
|
118
|
+
'}' +
|
119
|
+
'.chromeFrameOverlayCloseBar {' +
|
120
|
+
'height: 1em;' +
|
121
|
+
'text-align: right;' +
|
122
|
+
'background-color: #CADEF4;' +
|
123
|
+
'}' +
|
124
|
+
'.chromeFrameOverlayUnderlay {' +
|
125
|
+
'position: absolute;' +
|
126
|
+
'width: 100%;' +
|
127
|
+
'height: 100%;' +
|
128
|
+
'background-color: white;' +
|
129
|
+
'opacity: 0.5;' +
|
130
|
+
'-moz-opacity: 0.5;' +
|
131
|
+
'-webkit-opacity: 0.5;' +
|
132
|
+
'-ms-filter: ' +
|
133
|
+
'"progid:DXImageTransform.Microsoft.Alpha(Opacity=50)";' +
|
134
|
+
'filter: alpha(opacity=50);' +
|
135
|
+
'z-index: 2000;' +
|
136
|
+
'}';
|
137
|
+
injectStyleSheet(rules);
|
138
|
+
cfStyleTagInjected = true;
|
139
|
+
};
|
140
|
+
|
141
|
+
/**
|
142
|
+
* Injects style rules to hide the overlay version of the GCF prompt.
|
143
|
+
* Multiple calls have no effect.
|
144
|
+
*/
|
145
|
+
var closeOverlay = function() {
|
146
|
+
// IE has a limit to the # of <style> tags allowed, so we avoid
|
147
|
+
// tempting the fates.
|
148
|
+
if (cfHiddenInjected) {
|
149
|
+
return;
|
150
|
+
}
|
151
|
+
var rules = '.chromeFrameOverlayContent { display: none; }' +
|
152
|
+
'.chromeFrameOverlayUnderlay { display: none; }';
|
153
|
+
injectStyleSheet(rules);
|
154
|
+
// Hide the dialog for a year (or until cookies are deleted).
|
155
|
+
var age = 365 * 24 * 60 * 60 * 1000;
|
156
|
+
document.cookie = "disableGCFCheck=1;path=/;max-age="+age;
|
157
|
+
cfHiddenInjected = true;
|
158
|
+
};
|
159
|
+
|
160
|
+
/**
|
161
|
+
* Plucks properties from the passed arguments and sets them on the passed
|
162
|
+
* DOM node
|
163
|
+
* @param {Node} node The node to set properties on
|
164
|
+
* @param {Object} args A map of user-specified properties to set
|
165
|
+
*/
|
166
|
+
var setProperties = function(node, args) {
|
167
|
+
|
168
|
+
var srcNode = byId(args['node']);
|
169
|
+
|
170
|
+
node.id = args['id'] || (srcNode ? srcNode['id'] || getUid(srcNode) : '');
|
171
|
+
|
172
|
+
// TODO(slightlyoff): Opera compat? need to test there
|
173
|
+
var cssText = args['cssText'] || '';
|
174
|
+
node.style.cssText = ' ' + cssText;
|
175
|
+
|
176
|
+
var classText = args['className'] || '';
|
177
|
+
node.className = classText;
|
178
|
+
|
179
|
+
// default if the browser doesn't so we don't show sad-tab
|
180
|
+
var src = args['src'] || 'about:blank';
|
181
|
+
|
182
|
+
node.src = src;
|
183
|
+
|
184
|
+
if (srcNode) {
|
185
|
+
srcNode.parentNode.replaceChild(node, srcNode);
|
186
|
+
}
|
187
|
+
};
|
188
|
+
|
189
|
+
/**
|
190
|
+
* Creates an iframe.
|
191
|
+
* @param {Object} args A bag of configuration properties, including values
|
192
|
+
* like 'node', 'cssText', 'className', 'id', 'src', etc.
|
193
|
+
* @return {Node}
|
194
|
+
*/
|
195
|
+
var makeIframe = function(args) {
|
196
|
+
var el = document.createElement('iframe');
|
197
|
+
el.setAttribute('frameborder', '0');
|
198
|
+
el.setAttribute('border', '0');
|
199
|
+
setProperties(el, args);
|
200
|
+
return el;
|
201
|
+
};
|
202
|
+
|
203
|
+
/**
|
204
|
+
* Adds an unadorned iframe into the page, taking arguments to customize it.
|
205
|
+
* @param {Object} args A map of user-specified properties to set
|
206
|
+
*/
|
207
|
+
var makeInlinePrompt = function(args) {
|
208
|
+
args.className = 'chromeFrameInstallDefaultStyle ' +
|
209
|
+
(args.className || '');
|
210
|
+
var ifr = makeIframe(args);
|
211
|
+
// TODO(slightlyoff): handle placement more elegantly!
|
212
|
+
if (!ifr.parentNode) {
|
213
|
+
var firstChild = document.body.firstChild;
|
214
|
+
document.body.insertBefore(ifr, firstChild);
|
215
|
+
}
|
216
|
+
};
|
217
|
+
|
218
|
+
/**
|
219
|
+
* Adds a styled, closable iframe into the page with a background that
|
220
|
+
* emulates a modal dialog.
|
221
|
+
* @param {Object} args A map of user-specified properties to set
|
222
|
+
*/
|
223
|
+
var makeOverlayPrompt = function(args) {
|
224
|
+
if (byId('chromeFrameOverlayContent')) {
|
225
|
+
return; // Was previously created. Bail.
|
226
|
+
}
|
227
|
+
|
228
|
+
var n = document.createElement('span');
|
229
|
+
n.innerHTML = '<div class="chromeFrameOverlayUnderlay"></div>' +
|
230
|
+
'<table class="chromeFrameOverlayContent"' +
|
231
|
+
'id="chromeFrameOverlayContent"' +
|
232
|
+
'cellpadding="0" cellspacing="0">' +
|
233
|
+
'<tr class="chromeFrameOverlayCloseBar">' +
|
234
|
+
'<td>' +
|
235
|
+
// TODO(slightlyoff): i18n
|
236
|
+
'<button id="chromeFrameCloseButton">close</button>' +
|
237
|
+
'</td>' +
|
238
|
+
'</tr>' +
|
239
|
+
'<tr>' +
|
240
|
+
'<td id="chromeFrameIframeHolder"></td>' +
|
241
|
+
'</tr>' +
|
242
|
+
'</table>';
|
243
|
+
|
244
|
+
var b = document.body;
|
245
|
+
// Insert underlay nodes into the document in the right order.
|
246
|
+
while (n.firstChild) {
|
247
|
+
b.insertBefore(n.lastChild, b.firstChild);
|
248
|
+
}
|
249
|
+
var ifr = makeIframe(args);
|
250
|
+
byId('chromeFrameIframeHolder').appendChild(ifr);
|
251
|
+
byId('chromeFrameCloseButton').onclick = closeOverlay;
|
252
|
+
};
|
253
|
+
|
254
|
+
var CFInstall = {};
|
255
|
+
|
256
|
+
/**
|
257
|
+
* Checks to see if Chrome Frame is available, if not, prompts the user to
|
258
|
+
* install. Once installation is begun, a background timer starts,
|
259
|
+
* checkinging for a successful install every 2 seconds. Upon detection of
|
260
|
+
* successful installation, the current page is reloaded, or if a
|
261
|
+
* 'destination' parameter is passed, the page navigates there instead.
|
262
|
+
* @param {Object} args A bag of configuration properties. Respected
|
263
|
+
* properties are: 'mode', 'url', 'destination', 'node', 'onmissing',
|
264
|
+
* 'preventPrompt', 'oninstall', 'preventInstallDetection', 'cssText', and
|
265
|
+
* 'className'.
|
266
|
+
* @public
|
267
|
+
*/
|
268
|
+
CFInstall.check = function(args) {
|
269
|
+
args = args || {};
|
270
|
+
|
271
|
+
// We currently only support CF in IE
|
272
|
+
// TODO(slightlyoff): Update this should we support other browsers!
|
273
|
+
var ua = navigator.userAgent;
|
274
|
+
var ieRe = /MSIE (\S+); Windows NT/;
|
275
|
+
var bail = false;
|
276
|
+
if (ieRe.test(ua)) {
|
277
|
+
// We also only support Win2003/XPSP2 or better. See:
|
278
|
+
// http://msdn.microsoft.com/en-us/library/ms537503%28VS.85%29.aspx
|
279
|
+
if (parseFloat(ieRe.exec(ua)[1]) < 6 &&
|
280
|
+
// 'SV1' indicates SP2, only bail if not SP2 or Win2K3
|
281
|
+
ua.indexOf('SV1') < 0) {
|
282
|
+
bail = true;
|
283
|
+
}
|
284
|
+
} else {
|
285
|
+
// Not IE
|
286
|
+
bail = true;
|
287
|
+
}
|
288
|
+
if (bail) {
|
289
|
+
return;
|
290
|
+
}
|
291
|
+
|
292
|
+
// Inject the default styles
|
293
|
+
injectCFStyleTag();
|
294
|
+
|
295
|
+
if (document.cookie.indexOf("disableGCFCheck=1") >=0) {
|
296
|
+
// If we're supposed to hide the overlay prompt, add the rules to do it.
|
297
|
+
closeOverlay();
|
298
|
+
}
|
299
|
+
|
300
|
+
// When loaded in an alternate protocol (e.g., "file:"), still call out to
|
301
|
+
// the right location.
|
302
|
+
var currentProtocol = document.location.protocol;
|
303
|
+
var protocol = (currentProtocol == 'https:') ? 'https:' : 'http:';
|
304
|
+
// TODO(slightlyoff): Update this URL when a mini-installer page is
|
305
|
+
// available.
|
306
|
+
var installUrl = protocol + '//www.google.com/chromeframe';
|
307
|
+
if (!isAvailable()) {
|
308
|
+
if (args.onmissing) {
|
309
|
+
args.onmissing();
|
310
|
+
}
|
311
|
+
|
312
|
+
args.src = args.url || installUrl;
|
313
|
+
var mode = args.mode || 'inline';
|
314
|
+
var preventPrompt = args.preventPrompt || false;
|
315
|
+
|
316
|
+
if (!preventPrompt) {
|
317
|
+
if (mode == 'inline') {
|
318
|
+
makeInlinePrompt(args);
|
319
|
+
} else if (mode == 'overlay') {
|
320
|
+
makeOverlayPrompt(args);
|
321
|
+
} else {
|
322
|
+
window.open(args.src);
|
323
|
+
}
|
324
|
+
}
|
325
|
+
|
326
|
+
if (args.preventInstallDetection) {
|
327
|
+
return;
|
328
|
+
}
|
329
|
+
|
330
|
+
// Begin polling for install success.
|
331
|
+
var installTimer = setInterval(function() {
|
332
|
+
// every 2 seconds, look to see if CF is available, if so, proceed on
|
333
|
+
// to our destination
|
334
|
+
if (isAvailable()) {
|
335
|
+
if (args.oninstall) {
|
336
|
+
args.oninstall();
|
337
|
+
}
|
338
|
+
|
339
|
+
clearInterval(installTimer);
|
340
|
+
// TODO(slightlyoff): add a way to prevent navigation or make it
|
341
|
+
// contingent on oninstall?
|
342
|
+
window.location = args.destination || window.location;
|
343
|
+
}
|
344
|
+
}, 2000);
|
345
|
+
}
|
346
|
+
};
|
347
|
+
|
348
|
+
CFInstall._force = false;
|
349
|
+
CFInstall._forceValue = false;
|
350
|
+
CFInstall.isAvailable = isAvailable;
|
351
|
+
|
352
|
+
// expose CFInstall to the external scope. We've already checked to make
|
353
|
+
// sure we're not going to blow existing objects away.
|
354
|
+
scope.CFInstall = CFInstall;
|
355
|
+
|
356
|
+
})(this['ChromeFrameInstallScope'] || this);
|