summaryrefslogtreecommitdiffabout
authorwilliamt <williamt>2007-01-05 10:06:01 (UTC)
committer williamt <williamt>2007-01-05 10:06:01 (UTC)
commitc4b3096e166a0a9a60fd27c8f2dc06e688f13172 (patch) (unidiff)
treee5f234745835e28d56e5d5b7d6146886d04b3436
downloadfoxri-c4b3096e166a0a9a60fd27c8f2dc06e688f13172.zip
foxri-c4b3096e166a0a9a60fd27c8f2dc06e688f13172.tar.gz
foxri-c4b3096e166a0a9a60fd27c8f2dc06e688f13172.tar.bz2
Initial revision
Diffstat (more/less context) (ignore whitespace changes)
-rwxr-xr-xmisc/xrdsContentHandler.js401
-rwxr-xr-xsrc/chrome.manifest2
-rwxr-xr-xsrc/chrome/content/aim_logo.gifbin0 -> 1247 bytes
-rwxr-xr-xsrc/chrome/content/email_icon.gifbin0 -> 476 bytes
-rwxr-xr-xsrc/chrome/content/feed-icon-14x14.pngbin0 -> 689 bytes
-rwxr-xr-xsrc/chrome/content/feed-icon-28x28.pngbin0 -> 1737 bytes
-rwxr-xr-xsrc/chrome/content/foxri.xul32
-rwxr-xr-xsrc/chrome/content/foxri_explorer.css124
-rwxr-xr-xsrc/chrome/content/i-contact_logo.gifbin0 -> 2391 bytes
-rwxr-xr-xsrc/chrome/content/i-forwarding_logo.gifbin0 -> 1293 bytes
-rwxr-xr-xsrc/chrome/content/i-sso_logo.gifbin0 -> 1917 bytes
-rwxr-xr-xsrc/chrome/content/icn_sso_lg.gifbin0 -> 1917 bytes
-rwxr-xr-xsrc/chrome/content/jabber_logo.gifbin0 -> 1199 bytes
-rwxr-xr-xsrc/chrome/content/msn_logo.gifbin0 -> 1190 bytes
-rwxr-xr-xsrc/chrome/content/openid_logo.pngbin0 -> 2489 bytes
-rwxr-xr-xsrc/chrome/content/skype_add_large.pngbin0 -> 2928 bytes
-rwxr-xr-xsrc/chrome/content/skype_call.pngbin0 -> 910 bytes
-rwxr-xr-xsrc/chrome/content/skype_call_large.pngbin0 -> 2348 bytes
-rwxr-xr-xsrc/chrome/content/skype_chat_large.pngbin0 -> 2721 bytes
-rwxr-xr-xsrc/chrome/content/skype_logo.pngbin0 -> 4331 bytes
-rwxr-xr-xsrc/chrome/content/xrdsHandler.js258
-rwxr-xr-xsrc/chrome/content/yahoo_logo.gifbin0 -> 1292 bytes
-rwxr-xr-xsrc/components/xriProtocolHandler.js1060
-rwxr-xr-xsrc/install.rdf40
24 files changed, 1917 insertions, 0 deletions
diff --git a/misc/xrdsContentHandler.js b/misc/xrdsContentHandler.js
new file mode 100755
index 0000000..4f30310
--- a/dev/null
+++ b/misc/xrdsContentHandler.js
@@ -0,0 +1,401 @@
1
2const nsISupports = Components.interfaces.nsISupports;
3
4const nsIBrowserDOMWindow = Components.interfaces.nsIBrowserDOMWindow;
5const nsIChannel = Components.interfaces.nsIChannel;
6const nsIContentHandler = Components.interfaces.nsIContentHandler;
7const nsIDOMChromeWindow = Components.interfaces.nsIDOMChromeWindow;
8const nsIDOMWindow = Components.interfaces.nsIDOMWindow;
9const nsIFactory = Components.interfaces.nsIFactory;
10const nsIHttpProtocolHandler = Components.interfaces.nsIHttpProtocolHandler;
11const nsIInterfaceRequestor = Components.interfaces.nsIInterfaceRequestor;
12const nsIPrefBranch = Components.interfaces.nsIPrefBranch;
13const nsIPrefLocalizedString = Components.interfaces.nsIPrefLocalizedString;
14const nsISupportsString = Components.interfaces.nsISupportsString;
15const nsIWebNavigation = Components.interfaces.nsIWebNavigation;
16const nsIWindowMediator = Components.interfaces.nsIWindowMediator;
17const nsIWindowWatcher = Components.interfaces.nsIWindowWatcher;
18const nsICategoryManager = Components.interfaces.nsICategoryManager;
19const nsIWebNavigationInfo = Components.interfaces.nsIWebNavigationInfo;
20const nsIStreamListener = Components.interfaces.nsIStreamListener;
21const nsIRequestObserver = Components.interfaces.nsIRequestObserver;
22const nsIURILoader = Components.interfaces.nsIURILoader;
23const nsIURIContentListener = Components.interfaces.nsIURIContentListener;
24
25const NS_BINDING_ABORTED = 0x80020006;
26const NS_ERROR_WONT_HANDLE_CONTENT = 0x805d0001;
27const NS_ERROR_ABORT = Components.results.NS_ERROR_ABORT;
28
29function shouldLoadURI(aURI) {
30 if (aURI.schemeIs("xri"))
31 return true;
32 dump("*** Preventing external load of chrome: URI into browser window\n");
33 dump(" Use -chrome <uri> instead\n");
34 return false;
35}
36
37
38
39
40function openWindow(parent, url, target, features, args) {
41 var wwatch = Components.classes["@mozilla.org/embedcomp/window-watcher;1"]
42 .getService(nsIWindowWatcher);
43
44 var argstring;
45 if (args) {
46 argstring = Components.classes["@mozilla.org/supports-string;1"]
47 .createInstance(nsISupportsString);
48 argstring.data = args;
49 }
50 return wwatch.openWindow(parent, url, target, features, argstring);
51}
52
53
54
55
56function getMostRecentWindow(aType) {
57 var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
58 .getService(nsIWindowMediator);
59 return wm.getMostRecentWindow(aType);
60}
61
62
63
64
65// this returns the most recent non-popup browser window
66function getMostRecentBrowserWindow() {
67 var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
68 .getService(Components.interfaces.nsIWindowMediator);
69
70 var windowList = wm.getZOrderDOMWindowEnumerator("navigator:browser", true);
71 if (!windowList.hasMoreElements())
72 return null;
73
74 var win = windowList.getNext();
75 while (!win.toolbar.visible) {
76 if (!windowList.hasMoreElements())
77 return null;
78
79 win = windowList.getNext();
80 }
81
82 return win;
83}
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102var xrdsContentHandler = {
103
104 /** browser window */
105 contentWindow: null,
106
107 /** XRDS buffer */
108 buf: null,
109
110 scriptableInStream: null,
111
112 init: function(contentWin) {
113 this.contentWindow = contentWindow;
114 },
115
116 close: function() {
117 this.contentWindow = null;
118 },
119
120 /* nsISupports */
121 QueryInterface : function(iid) {
122 dump("xrdsCH ... ");
123 if (iid.equals(nsISupports)) {
124 dump("QI(nsISupports)\n");
125 }
126 else if (iid.equals(nsIContentHandler)) {
127 dump("QI(nsIContentHandler)\n");
128 }
129 else if (iid.equals(nsIStreamListener)) {
130 dump("QI(nsIStreamListener)\n");
131 }
132 else if (iid.equals(nsIRequestObserver)) {
133 dump("QI(nsIRequestObserver)\n");
134 }
135 else if (iid.equals(Components.interfaces.nsIURIContentListener)) {
136 dump("QI(nsIURIContentListener)\n");
137 }
138 else if (iid.equals(nsIFactory)) {
139 dump("QI(nsIFactory)\n");
140 }
141 else {
142 dump("QI(" + iid + ") - IDONTKNOW!!!!!\n");
143 }
144
145 if (!iid.equals(nsISupports) &&
146 !iid.equals(nsIContentHandler) &&
147 !iid.equals(nsIURIContentListener) &&
148 !iid.equals(nsIStreamListener) &&
149 !iid.equals(nsIRequestObserver) &&
150 !iid.equals(nsIFactory))
151 throw Components.results.NS_ERROR_NO_INTERFACE;
152
153 dump("QI returning this..\n");
154 return this;
155 },
156
157
158 /* nsIURIContentListener */
159 loadCookie: null,
160
161 parentContentListener: null,
162
163 onStartURIOpen: function(uri)
164 {
165 dump("xrdsCH onStartURIOpen '" + uri + "'");
166 // ignore and don't abort
167 return false;
168 },
169
170 doContent: function(contentType, isContentPreferred, request, contentHandler)
171 {
172 dump("doContent called\n");
173 // forward the doContent to our content area webshell
174 var docShell = this.contentWindow.docShell;
175 var contentListener;
176 try {
177 contentListener =
178 docShell.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
179 .getInterface(Components.interfaces.nsIURIContentListener);
180 } catch (ex) {
181 dump(ex);
182 }
183
184 dump("no content listener from docShell!\n");
185 if (!contentListener) return false;
186
187 var rv = contentListener.doContent(contentType, isContentPreferred, request, contentHandler);
188
189 if (rv) {
190 dump("docShell wants to handle the load..\n");
191 }
192 else {
193 dump("docShell does NOT want to handle the load..\n");
194
195 contentHandler = contentHandler.value;
196 }
197
198 return rv;
199
200 },
201
202 isPreferred: function(contentType, desiredContentType)
203 {
204 dump("isPreferred '" + contentType + "'\n");
205 switch(contentType) {
206 case "application/xrds+xml":
207 case "application/xrd+xml":
208 dump("yes!!!\n");
209 return true;
210 }
211 dump("erm.. nope!\n");
212 return false;
213 },
214
215 canHandleContent: function(contentType, isContentPreferred, desiredContentType)
216 {
217 dump("canHandleContent '" + contentType + "'\n");
218 return this.isPreferred(contentType, desiredContentType);
219 },
220
221
222
223 /* nsIRequestObserver */
224 onStartRequest: function(request, ctx)
225 {
226 dump("xrdsContentHandler - onStartRequest\n");
227 },
228
229 onStopRequest: function(request, ctx, status)
230 {
231 dump("xrdsContentHandler - onStopRequest\n");
232 this.scriptableInStream.close();
233 },
234
235
236 /* nsIStreamListener */
237 onDataAvailable: function(request, domWindow, inputStream, offset, count)
238 {
239 dump("onDataAvailable, offset=" + offset + ", count=" + count + "\n");
240 if (offset == 0) {
241 this.scriptableInStream.init(inputStream);
242 }
243
244 buf += this.scriptableInStream.read(count);
245
246 if (!request.isPending()) {
247 dump("request finished, buf = " + buf + "\n");
248 var html = domWindow.document.createElement("html");
249 html.createTextNode(buf);
250 this.scriptableInStream = null;
251 }
252 else {
253 dump("request pending...\n");
254 }
255 },
256
257 /* nsIContentHandler */
258
259 handleContent : function(contentType, context, request)
260 {
261 dump("handleContent " + contentType + "\n");
262 var parentWin;
263 try {
264 parentWin = context.getInterface(nsIDOMWindow);
265 }
266 catch (e) {
267 alert("no parent!!!"); // XXX
268 return;
269 }
270
271 dump("getting channel\n");
272 var channel = request.QueryInterface(nsIChannel);
273 if (!channel) {
274 dump("no channel!!!\n");
275 return;
276 }
277
278 if (this.scriptableInStream) {
279 dump("Hey! You can't possibly be reusing this handler?!\n");
280 return;
281 }
282
283 dump("making scriptableInStream\n");
284 this.scriptableInStream = Components.classes["@mozilla.org/scriptableinputstream;1"]
285 .createInstance(Components.interfaces.nsIScriptableInputStream);
286
287 buf = '';
288
289/*
290 try {
291 dump("before asyncOpen\n");
292 channel.asyncOpen(this, parentWin);
293 return;
294 }
295 catch (e) {
296 dump("shit cannot asyncOpen... " + e + "\n");
297 }
298
299 dump("try blocking open\n");
300 var instream = channel.open();
301 this.scriptableinputstream.init(instream);
302 while (true) {
303 var n = this.scriptableinputstream.available();
304 if (n > 0) {
305 dump(this.scriptableinputstream.read(n) + "\n");
306 }
307 }
308
309 dump("channel opened\n");
310*/
311
312 dump("handleContent returning \n");
313 },
314
315 /* nsIFactory */
316 createInstance: function(outer, iid)
317 {
318 dump("xrdsCH createInstance called..\n");
319 if (outer != null)
320 throw Components.results.NS_ERROR_NO_AGGREGATION;
321
322 dump("xrdsCH ok\n");
323 return this.QueryInterface(iid);
324 },
325
326 lockFactory : function bch_lock(lock) {
327 /* no-op */
328 }
329};
330
331const CLASS_ID = Components.ID("{e857f11b-fd59-4285-8f4a-c27cf9051a36}");
332const CLASS_NAME = "XRDS Content Handler";
333const CONTRACTID_PREFIX = "@mozilla.org/uriloader/content-handler;1?type=";
334
335
336
337var Module = {
338 /* nsISupports */
339 QueryInterface: function(iid)
340 {
341 dump("QI called\n");
342 if (iid.equals(Components.interfaces.nsIModule) ||
343 iid.equals(Components.interfaces.nsISupports))
344 return this;
345
346 dump("QI throwing!!\n");
347 throw Components.results.NS_ERROR_NO_INTERFACE;
348 },
349
350
351 /* nsIModule */
352 getClassObject: function(compMgr, cid, iid)
353 {
354 dump("getClassObject called\n");
355 if (cid.equals(CLASS_ID))
356 return xrdsContentHandler.QueryInterface(iid);
357
358 dump("gCO throwing!!\n");
359 throw Components.results.NS_ERROR_NO_INTERFACE;
360 },
361
362
363 registerSelf: function(compMgr, fileSpec, location, type)
364 {
365 var compReg =
366 compMgr.QueryInterface(Components.interfaces.nsIComponentRegistrar);
367
368 compReg.registerFactoryLocation( CLASS_ID,
369 CLASS_NAME,
370 CONTRACTID_PREFIX + "application/xrds+xml",
371 fileSpec,
372 location,
373 type );
374
375 compReg.registerFactoryLocation( CLASS_ID,
376 CLASS_NAME,
377 CONTRACTID_PREFIX + "text/uri-list",
378 fileSpec,
379 location,
380 type );
381
382 dump("done registerSelf..\n");
383 },
384
385
386 unregisterSelf: function(compMgr, location, type)
387 {
388 var compReg = compMgr.QueryInterface(nsIComponentRegistrar);
389 compReg.unregisterFactoryLocation(CLASS_ID, location);
390 },
391
392
393 canUnload: function(compMgr) {
394 return true;
395 }
396};
397
398// NSGetModule: Return the nsIModule object.
399function NSGetModule(compMgr, fileSpec) {
400 return Module;
401}
diff --git a/src/chrome.manifest b/src/chrome.manifest
new file mode 100755
index 0000000..8ff2425
--- a/dev/null
+++ b/src/chrome.manifest
@@ -0,0 +1,2 @@
1 content foxrichrome/content/
2 overlay chrome://browser/content/browser.xulchrome://foxri/content/foxri.xul
diff --git a/src/chrome/content/aim_logo.gif b/src/chrome/content/aim_logo.gif
new file mode 100755
index 0000000..9d5bffe
--- a/dev/null
+++ b/src/chrome/content/aim_logo.gif
Binary files differ
diff --git a/src/chrome/content/email_icon.gif b/src/chrome/content/email_icon.gif
new file mode 100755
index 0000000..1c62b57
--- a/dev/null
+++ b/src/chrome/content/email_icon.gif
Binary files differ
diff --git a/src/chrome/content/feed-icon-14x14.png b/src/chrome/content/feed-icon-14x14.png
new file mode 100755
index 0000000..b3c949d
--- a/dev/null
+++ b/src/chrome/content/feed-icon-14x14.png
Binary files differ
diff --git a/src/chrome/content/feed-icon-28x28.png b/src/chrome/content/feed-icon-28x28.png
new file mode 100755
index 0000000..d64c669
--- a/dev/null
+++ b/src/chrome/content/feed-icon-28x28.png
Binary files differ
diff --git a/src/chrome/content/foxri.xul b/src/chrome/content/foxri.xul
new file mode 100755
index 0000000..dc1cc37
--- a/dev/null
+++ b/src/chrome/content/foxri.xul
@@ -0,0 +1,32 @@
1<?xml version="1.0"?>
2<?xml-stylesheet href="foxri.css" type="text/css"?>
3
4
5<overlay id="foxri" xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">
6
7
8<!--
9<script type="application/x-javascript" src="chrome://foxri/content/xrdsHandler.js"/>
10-->
11
12<script>
13function xriFix(event)
14{
15 var txt = gURLBar.value;
16 if (/^(=|@|!)\S+/.test(txt)) {
17 gURLBar.value = "xri://" + txt;
18 }
19}
20</script>
21
22
23 <toolbaritem id="urlbar-container">
24 <textbox id="urlbar" ontextentered="xriFix(); return handleURLBarCommand(param);" />
25 </toolbaritem>
26
27 <toolbaritem id="go-container">
28 <toolbarbutton id="go-button" oncommand="xriFix(); return handleURLBarCommand(event);" />
29 </toolbaritem>
30
31</overlay>
32
diff --git a/src/chrome/content/foxri_explorer.css b/src/chrome/content/foxri_explorer.css
new file mode 100755
index 0000000..9e68e95
--- a/dev/null
+++ b/src/chrome/content/foxri_explorer.css
@@ -0,0 +1,124 @@
1
2
3* {
4 margin: 0;
5 padding: 0;
6}
7
8
9body {
10 background-color: #b1d8ff;
11 font-family: Trebuchet MS, Verdana, Helvetica, Arial, sans-serif;
12}
13
14
15a {
16 text-decoration: none;
17}
18
19a:link, a:visited {
20 color: #D65B5B; /* inames.net logo color */
21}
22
23a:hover {
24 border-bottom: 1px dashed #555A5C;
25}
26
27a img {
28 border: none;
29 text-decoration: none;
30}
31
32
33h1,h2,h3,h4,h5,h6,p {
34 margin-left: 100px;
35}
36
37h1 {
38 margin: 0 100px;
39 padding: 0;
40 color: #D65B5B; /* inames.net logo color */
41}
42
43h3 {
44 margin: 0 100px;
45 padding: 0;
46 color: #666666;
47}
48
49h3 strong {
50 color: #FF0000; /* inames.net logo color */
51}
52
53
54.service {
55 background-color: #ffffff; /* inames.net */
56 color: #333333;
57
58 margin: 20px 100px;
59 padding: 10px 150px;
60 min-height: 100px;
61 background-repeat: no-repeat;
62 background-position: 15px 10px;
63
64 -moz-border-radius: 10px;
65}
66
67
68.srv_skype {
69 background-image: url(chrome://foxri/content/skype_logo.png);
70}
71
72.srv_aim {
73 background-image: url(chrome://foxri/content/aim_logo.gif);
74}
75
76.srv_yahoo {
77 background-image: url(chrome://foxri/content/yahoo_logo.gif);
78}
79
80.srv_msn {
81 background-image: url(chrome://foxri/content/msn_logo.gif);
82}
83
84.srv_jabber {
85 background-image: url(chrome://foxri/content/jabber_logo.gif);
86}
87
88.srv_openid {
89 background-image: url(chrome://foxri/content/openid_logo.png);
90}
91
92.srv_i-contact {
93 background-image: url(chrome://foxri/content/i-contact_logo.gif);
94}
95
96.srv_i-forwarding {
97 background-image: url(chrome://foxri/content/i-forwarding_logo.gif);
98}
99
100.srv_authn-saml {
101 background-image: url(chrome://foxri/content/i-sso_logo.gif);
102}
103
104.srv_email {
105 background-image: url(chrome://foxri/content/email_icon.gif);
106}
107
108.srv_feed {
109 background-image: url(chrome://foxri/content/feed-icon-28x28.png);
110}
111
112.service a {
113 color: #D65B5B; /* inames.net logo color */
114}
115
116
117div a:hover {
118 text-decoration: none;
119 border: none;
120}
121
122.error {
123 color: #FF5B5B;
124}
diff --git a/src/chrome/content/i-contact_logo.gif b/src/chrome/content/i-contact_logo.gif
new file mode 100755
index 0000000..9775919
--- a/dev/null
+++ b/src/chrome/content/i-contact_logo.gif
Binary files differ
diff --git a/src/chrome/content/i-forwarding_logo.gif b/src/chrome/content/i-forwarding_logo.gif
new file mode 100755
index 0000000..125b1de
--- a/dev/null
+++ b/src/chrome/content/i-forwarding_logo.gif
Binary files differ
diff --git a/src/chrome/content/i-sso_logo.gif b/src/chrome/content/i-sso_logo.gif
new file mode 100755
index 0000000..f948dd6
--- a/dev/null
+++ b/src/chrome/content/i-sso_logo.gif
Binary files differ
diff --git a/src/chrome/content/icn_sso_lg.gif b/src/chrome/content/icn_sso_lg.gif
new file mode 100755
index 0000000..f948dd6
--- a/dev/null
+++ b/src/chrome/content/icn_sso_lg.gif
Binary files differ
diff --git a/src/chrome/content/jabber_logo.gif b/src/chrome/content/jabber_logo.gif
new file mode 100755
index 0000000..01628d1
--- a/dev/null
+++ b/src/chrome/content/jabber_logo.gif
Binary files differ
diff --git a/src/chrome/content/msn_logo.gif b/src/chrome/content/msn_logo.gif
new file mode 100755
index 0000000..19dd647
--- a/dev/null
+++ b/src/chrome/content/msn_logo.gif
Binary files differ
diff --git a/src/chrome/content/openid_logo.png b/src/chrome/content/openid_logo.png
new file mode 100755
index 0000000..51d3654
--- a/dev/null
+++ b/src/chrome/content/openid_logo.png
Binary files differ
diff --git a/src/chrome/content/skype_add_large.png b/src/chrome/content/skype_add_large.png
new file mode 100755
index 0000000..2cfb5dc
--- a/dev/null
+++ b/src/chrome/content/skype_add_large.png
Binary files differ
diff --git a/src/chrome/content/skype_call.png b/src/chrome/content/skype_call.png
new file mode 100755
index 0000000..bbb9807
--- a/dev/null
+++ b/src/chrome/content/skype_call.png
Binary files differ
diff --git a/src/chrome/content/skype_call_large.png b/src/chrome/content/skype_call_large.png
new file mode 100755
index 0000000..65ce0dd
--- a/dev/null
+++ b/src/chrome/content/skype_call_large.png
Binary files differ
diff --git a/src/chrome/content/skype_chat_large.png b/src/chrome/content/skype_chat_large.png
new file mode 100755
index 0000000..9642d8a
--- a/dev/null
+++ b/src/chrome/content/skype_chat_large.png
Binary files differ
diff --git a/src/chrome/content/skype_logo.png b/src/chrome/content/skype_logo.png
new file mode 100755
index 0000000..a190417
--- a/dev/null
+++ b/src/chrome/content/skype_logo.png
Binary files differ
diff --git a/src/chrome/content/xrdsHandler.js b/src/chrome/content/xrdsHandler.js
new file mode 100755
index 0000000..d4b0d35
--- a/dev/null
+++ b/src/chrome/content/xrdsHandler.js
@@ -0,0 +1,258 @@
1
2const nsISupports = Components.interfaces.nsISupports;
3const nsIChannel = Components.interfaces.nsIChannel;
4const nsIContentHandler = Components.interfaces.nsIContentHandler;
5const nsIDOMWindow = Components.interfaces.nsIDOMWindow;
6const nsIInterfaceRequestor = Components.interfaces.nsIInterfaceRequestor;
7const nsIStreamListener = Components.interfaces.nsIStreamListener;
8const nsIRequestObserver = Components.interfaces.nsIRequestObserver;
9const nsIURILoader = Components.interfaces.nsIURILoader;
10const nsIURIContentListener = Components.interfaces.nsIURIContentListener;
11
12
13
14
15var g_xrdsHandler = null;
16
17
18
19
20
21
22XrdsContentHandler.prototype = {
23
24 /** browser window */
25 contentWindow: null,
26
27 /** XRDS buffer */
28 buf: null,
29
30 scriptableInStream: null,
31
32 init: function(contentWin) {
33 dump("XrdsContentHandler.init()\n");
34 this.contentWindow = contentWin;
35
36 var uriLoader = Components.classes["@mozilla.org/uriloader;1"]
37 .getService(Components.interfaces.nsIURILoader);
38 uriLoader.registerContentListener(this);
39 dump("XrdsContentHandler.init() returning\n");
40 },
41
42 close: function() {
43 this.contentWindow = null;
44 var uriLoader = Components.classes["@mozilla.org/uriloader;1"]
45 .getService(Components.interfaces.nsIURILoader);
46 uriLoader.unRegisterContentListener(g_xrdsHandler);
47 },
48
49 /* nsISupports */
50 QueryInterface : function(iid) {
51 dump("xrdsCH ... ");
52 if (iid.equals(nsISupports)) {
53 dump("QI(nsISupports)\n");
54 }
55 else if (iid.equals(nsIContentHandler)) {
56 dump("QI(nsIContentHandler)\n");
57 }
58 else if (iid.equals(nsIStreamListener)) {
59 dump("QI(nsIStreamListener)\n");
60 }
61 else if (iid.equals(nsIRequestObserver)) {
62 dump("QI(nsIRequestObserver)\n");
63 }
64 else if (iid.equals(Components.interfaces.nsIURIContentListener)) {
65 dump("QI(nsIURIContentListener)\n");
66 }
67 else if (iid.equals(nsIFactory)) {
68 dump("QI(nsIFactory)\n");
69 }
70 else {
71 dump("QI(" + iid + ") - IDONTKNOW!!!!!\n");
72 }
73
74 if (!iid.equals(nsISupports) &&
75// !iid.equals(nsIContentHandler) &&
76 !iid.equals(nsIURIContentListener) &&
77 !iid.equals(nsIStreamListener) &&
78// !iid.equals(nsIRequestObserver) &&
79 !iid.equals(nsIFactory))
80 throw Components.results.NS_ERROR_NO_INTERFACE;
81
82 dump("QI returning this..\n");
83 return this;
84 },
85
86
87 /* nsIURIContentListener */
88 loadCookie: null,
89
90 parentContentListener: null,
91
92 onStartURIOpen: function(uri)
93 {
94 dump("xrdsCH onStartURIOpen '" + uri + "'");
95 // ignore and don't abort
96 return false;
97 },
98
99 doContent: function(contentType, isContentPreferred, request, contentHandler)
100 {
101 dump("doContent called\n");
102 // forward the doContent to our content area webshell
103 var docShell = this.contentWindow.docShell;
104 var contentListener;
105 try {
106 contentListener =
107 docShell.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
108 .getInterface(Components.interfaces.nsIURIContentListener);
109 } catch (ex) {
110 dump(ex);
111 }
112
113 dump("no content listener from docShell!\n");
114 if (!contentListener) return false;
115
116 var rv = contentListener.doContent(contentType, isContentPreferred, request, contentHandler);
117
118 if (rv) {
119 dump("docShell wants to handle the load..\n");
120 }
121 else {
122 dump("docShell does NOT want to handle the load..\n");
123
124 contentHandler = contentHandler.value;
125 }
126
127 return rv;
128
129 },
130
131 isPreferred: function(contentType, desiredContentType)
132 {
133 dump("isPreferred '" + contentType + "'\n");
134 switch(contentType) {
135 case "application/xrds+xml":
136 case "application/xrd+xml":
137 dump("yes!!!\n");
138 return true;
139 }
140 dump("erm.. nope!\n");
141 return false;
142 },
143
144 canHandleContent: function(contentType, isContentPreferred, desiredContentType)
145 {
146 dump("canHandleContent '" + contentType + "'\n");
147 return this.isPreferred(contentType, desiredContentType);
148 },
149
150
151
152 /* nsIRequestObserver */
153 onStartRequest: function(request, ctx)
154 {
155 dump("xrdsContentHandler - onStartRequest\n");
156 },
157
158 onStopRequest: function(request, ctx, status)
159 {
160 dump("xrdsContentHandler - onStopRequest\n");
161 this.scriptableInStream.close();
162 },
163
164
165 /* nsIStreamListener */
166 onDataAvailable: function(request, domWindow, inputStream, offset, count)
167 {
168 dump("onDataAvailable, offset=" + offset + ", count=" + count + "\n");
169 if (offset == 0) {
170 this.scriptableInStream.init(inputStream);
171 }
172
173 buf += this.scriptableInStream.read(count);
174
175 if (!request.isPending()) {
176 dump("request finished, buf = " + buf + "\n");
177 var html = domWindow.document.createElement("html");
178 html.createTextNode(buf);
179 this.scriptableInStream = null;
180 }
181 else {
182 dump("request pending...\n");
183 }
184 },
185
186 /* nsIContentHandler */
187
188 handleContent : function(contentType, context, request)
189 {
190 dump("handleContent " + contentType + "\n");
191 var parentWin;
192 try {
193 parentWin = context.getInterface(nsIDOMWindow);
194 }
195 catch (e) {
196 alert("no parent!!!"); // XXX
197 return;
198 }
199
200 dump("getting channel\n");
201 var channel = request.QueryInterface(nsIChannel);
202 if (!channel) {
203 dump("no channel!!!\n");
204 return;
205 }
206
207 if (this.scriptableInStream) {
208 dump("Hey! You can't possibly be reusing this handler?!\n");
209 return;
210 }
211
212 dump("making scriptableInStream\n");
213 this.scriptableInStream = Components.classes["@mozilla.org/scriptableinputstream;1"]
214 .createInstance(Components.interfaces.nsIScriptableInputStream);
215
216 buf = '';
217
218 }
219
220};
221
222
223function XrdsContentHandler(contentWindow)
224{
225 this.init(contentWindow);
226}
227
228
229
230
231
232
233
234
235
236
237
238
239
240function foxri_startup() {
241 // our XRDS content-handler also does nsIURIContentListener
242 var xrdsHandler = new XrdsContentHandler(getBrowser());
243 g_xrdsHandler = xrdsHandler;
244
245}
246
247
248function foxri_shutdown() {
249 if (!g_xrdsHandler)
250 return;
251
252 g_xrdsHandler.close();
253}
254
255
256
257
258foxri_startup();
diff --git a/src/chrome/content/yahoo_logo.gif b/src/chrome/content/yahoo_logo.gif
new file mode 100755
index 0000000..34e6cd2
--- a/dev/null
+++ b/src/chrome/content/yahoo_logo.gif
Binary files differ
diff --git a/src/components/xriProtocolHandler.js b/src/components/xriProtocolHandler.js
new file mode 100755
index 0000000..7046cde
--- a/dev/null
+++ b/src/components/xriProtocolHandler.js
@@ -0,0 +1,1060 @@
1/***********************************************************
2 constants
3 ***********************************************************/
4
5// The interface we implement - nsIProtocolHandler
6const nsIProtocolHandler = Components.interfaces.nsIProtocolHandler;
7
8// Interfaces that we require
9 const nsISupports = Components.interfaces.nsISupports;
10 const nsIIOService = Components.interfaces.nsIIOService;
11 const nsIURI = Components.interfaces.nsIURI;
12 const nsIURL = Components.interfaces.nsIURL;
13 const nsIRequest = Components.interfaces.nsIRequest;
14const nsIRequestObserver = Components.interfaces.nsIRequestObserver;
15 const nsIChannel = Components.interfaces.nsIChannel;
16 const nsIHttpChannel = Components.interfaces.nsIHttpChannel;
17 const nsIStreamListener = Components.interfaces.nsIStreamListener;
18
19
20// UUID uniquely identifying our component
21// You can get from: http://kruithof.xs4all.nl/uuid/uuidgen here
22const CLASS_ID = Components.ID("{ea00b610-215a-11db-a98b-0800200c9a66}");
23
24// textual unique identifier
25const CONTRACT_ID = "@mozilla.org/network/protocol;1?name=xri";
26
27// Components that we require
28const CID_URI = "@mozilla.org/network/simple-uri;1";
29const kIOSERVICE_CID_STR = "{9ac9e770-18bc-11d3-9337-00104ba0fd40}";
30const CID_URL = "@mozilla.org/network/standard-url;1";
31
32// description
33const CLASS_NAME = "XRI Protocol Handler";
34
35const PROXY_URI = "http://xri.net/";
36
37const XP_ANY_TYPE = 0;
38const XP_NUMBER_TYPE = 1;
39const XP_STRING_TYPE = 2;
40const XP_BOOLEAN_TYPE = 3;
41const XP_UNORDERED_NODE_ITERATOR_TYPE = 4;
42const XP_ORDERED_NODE_ITERATOR_TYPE = 5;
43const XP_UNORDERED_NODE_SNAPSHOT_TYPE = 6;
44const XP_ORDERED_NODE_SNAPSHOT_TYPE = 7;
45const XP_ANY_UNORDERED_NODE_TYPE = 8;
46const XP_FIRST_ORDERED_NODE_TYPE = 9;
47
48
49var SERVICE_CLASSES = {
50 'xri://+i-service*(+contact)*($v*1.0)': 'i-contact',
51 'http://openid.net/signon/1.0': 'openid',
52 'xri://$res*auth*($v*2.0)': 'res-auth',
53 'xri://+i-service*(+authn)*(+saml)*($v*1.0)': 'authn-saml',
54 'xri://+i-service*(+metadata)*(+saml)*($v*1.0)' : 'metadata-saml',
55 'xri://+i-service*(+forwarding)*($v*1.0)': 'i-forwarding'
56};
57
58
59const HTML_HEAD = "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">\n\
60 <html xmlns=\"http://www.w3.org/1999/xhtml\">\n\
61 <head>\n\
62 <title>FoXRI Explorer - #QXRI#</title>\n\
63 <link href=\"chrome://foxri/content/foxri_explorer.css\" rel=\"stylesheet\" type=\"text/css\" />\n\
64 <body>\n\
65 <h1>FoXRI Explorer</h1>\n\
66 <div id=\"explorer_body\">\n";
67
68const HTML_FOOT = "</div>\n\
69 </body>\n\
70 </html>";
71
72
73
74/// Generic object method wrapper
75function methodWrapper(obj, method)
76{
77 return (
78 function() {
79 /* pass it this inner closure's arguments */
80 obj[method](arguments);
81 }
82 );
83}
84
85
86
87/// XRDS utility functions
88
89
90var nsResolver = {
91 lookupNamespaceURI: function(prefix)
92 {
93 if (prefix == "xrds")
94 return "xri://$xrds";
95 else if (prefix == "xrd")
96 return "xri://$xrd*($v*2.0)";
97 return "";
98 }
99};
100
101
102
103function runExpr(doc, context, expr, returnType)
104{
105 if (!returnType)
106 returnType = XP_ANY_TYPE;
107 var res = doc.evaluate(expr, context, nsResolver, returnType, null);
108 return res;
109}
110
111
112function getNumeric(doc, context, expr)
113{
114 var res = runExpr(doc, context, expr, XP_NUMBER_TYPE);
115 if (res)
116 return res.numberValue;
117 return null;
118}
119
120
121function getString(doc, context, expr)
122{
123 // var res = runExpr(doc, context, expr, XPathResult.STRING_TYPE);
124 var res = runExpr(doc, context, expr, XP_STRING_TYPE);
125 if (res)
126 return res.stringValue;
127 return null;
128}
129
130function getNode(doc, context, expr)
131{
132 var res = runExpr(doc, context, expr, XP_FIRST_ORDERED_NODE_TYPE);
133 if (res)
134 return res.singleNodeValue;
135 return null;
136}
137
138
139function getFinalXRD(doc)
140{
141 var lastNode = doc.firstChild;
142 while (true) {
143 var node = getNode(doc, lastNode, "xrds:XRDS[position()=last()]");
144 if (!node)
145 break;
146 lastNode = node;
147 }
148
149 return getNode(doc, lastNode, "xrd:XRD[position()=last()]");
150}
151
152
153function isIName(xri)
154{
155 if (xri.match('^xri://.!', 'i')) {
156 return false;
157 }
158 if (xri.match('^.!', 'i')) {
159 return false;
160 }
161 return true;
162}
163
164
165function arraySearch(a, re)
166{
167 var returnArr = new Array();
168 var i;
169 for (i = 0; i < a.length; i++) {
170 if (a[i].match(re)) {
171 returnArr.push(a[i]);
172 }
173 }
174
175 return returnArr;
176}
177
178
179function renderService(srv, doc, qxri)
180{
181 var html_types = '';
182 var html_paths = '';
183 var html_mediatypes = '';
184 var html_uris = '';
185 var html_actions = '';
186
187 var serviceName = friendlyServiceName(null);
188 var serviceType; // the last non-null Type
189 var knownServiceType; // first recognized service type
190
191 // get the types
192 var res = runExpr(doc, srv, "xrd:Type/text()");
193 var t;
194 while (t = res.iterateNext()) {
195 if (t.nodeValue) {
196 if (!knownServiceType && isKnownServiceType(t.nodeValue)) {
197 knownServiceType = t.nodeValue;
198 }
199
200 serviceType = t.nodeValue;
201 html_types += "<strong>Type:</strong> " + t.nodeValue + "<br/>";
202 }
203 }
204
205 // get the paths
206 res = runExpr(doc, srv, "xrd:Path/text()");
207 var p;
208 var qxri_prefix = qxri;
209 if (qxri_prefix.charAt(qxri_prefix.length - 1) != '/') {
210 qxri_prefix += '/';
211 }
212
213 while (p = res.iterateNext()) {
214 if (p.nodeValue) {
215 html_paths += "<strong>Path:</strong> " + p.nodeValue
216 + " [ <tt><a href=\"" + qxri_prefix + p.nodeValue + "\">"
217 + qxri_prefix + p.nodeValue + "</a></tt> ]"
218 + "<br/>\n";
219 }
220 }
221
222
223 // get the mediatypes
224 mediaTypes = new Array();
225 res = runExpr(doc, srv, "xrd:MediaType/text()");
226 var m;
227 while (m = res.iterateNext()) {
228 if (!knownServiceType) {
229 var srvType = guessServiceTypeByMime(m.nodeValue);
230 knownServiceType = srvType? srvType : null;
231 }
232
233 mediaTypes.push(m.nodeValue);
234 if (m.nodeValue) {
235 html_mediatypes += "<strong>Media Type:</strong> " + m.nodeValue + "<br/>";
236 }
237 }
238
239
240 res = runExpr(doc, srv, "xrd:URI/text()");
241 var u;
242 while (u = res.iterateNext()) {
243 if (!u.nodeValue)
244 continue;
245
246 var srvType = guessServiceTypeByURI(u.nodeValue);
247 if (!knownServiceType) {
248 knownServiceType = srvType;
249 }
250
251 html_uris += "<div class=\"" + getServiceClass(srvType) + "\">";
252
253 var linkContent = u.nodeValue;
254 var uriParts = u.nodeValue.match('^(.*):(.*)$');
255 if (!uriParts)
256 continue;
257
258 if (uriParts[1] == 'data') {
259 uriParts = uriParts[2].match('^(.*/.*),(.*)');
260 if (uriParts && uriParts[1].match('^image/', 'i')) {
261 linkContent = "<img src=\"" + u.nodeValue + "\"/>";
262 }
263 else if (uriParts) {
264 linkContent = uriParts[1] + " data";
265 }
266 }
267 else if (uriParts[1] == 'skype') {
268 uriParts = uriParts[2].match('^(.*)\\?(.*)');
269 if (uriParts) {
270 if (uriParts[2] == "call") {
271 linkContent = "<img src=\"chrome://foxri/content/skype_call_large.png\" alt=\"Call " + uriParts[1] + "\"/>";
272 }
273 else if (uriParts[2] == "chat") {
274 linkContent = "<img src=\"chrome://foxri/content/skype_chat_large.png\" alt=\"Chat with " + uriParts[1] + "\"/>";
275 }
276 else if (uriParts[2] == "add") {
277 linkContent = "<img src=\"chrome://foxri/content/skype_add_large.png\" alt=\"Add " + uriParts[1] + " to Skype\"/>";
278 }
279 }
280 }
281 else if (uriParts[1] == 'aim') {
282 uriParts = uriParts[2].match('^(.*)\\?.*screenname=([^&]*)', 'i');
283 if (uriParts) {
284 linkContent = "<img src=\"chrome://foxri/content/aim_logo.gif\" alt=\"Chat with " + uriParts[2] + "\"/> Chat with " + uriParts[2];
285 }
286 }
287
288 html_uris += "<a href=\""+u.nodeValue+"\">"
289 + linkContent + "</a>";
290 html_uris += "</div>";
291 }
292
293 var html = "<div class=\"service srv_" + getServiceClass(knownServiceType) + "\">\n";
294 html += html_types;
295 html += html_paths;
296 html += html_mediatypes;
297 if (html_uris) {
298 html += "<strong>URI(s):</strong><br/>\n";
299 html += html_uris;
300 }
301 html += "</div>";
302
303 return html;
304}
305
306
307
308function isKnownServiceType(type)
309{
310 if (type.toLowerCase() in SERVICE_CLASSES) {
311 return true;
312 }
313 return false;
314}
315
316function getServiceClass(type)
317{
318 if (isKnownServiceType(type)) {
319 return SERVICE_CLASSES[type.toLowerCase()];
320 }
321 return type;
322}
323
324
325function guessServiceTypeByURI(uri)
326{
327 if (uri == null || uri == "") {
328 return "unknown";
329 }
330 if (uri.match(/^https?:/i)) {
331 return "www";
332 }
333 else if (uri.match(/^skype:/i)) {
334 return "skype";
335 }
336 else if (uri.match(/^aim:/i)) {
337 return "aim";
338 }
339 else if (uri.match(/^xmpp:/i)) {
340 return "jabber";
341 }
342 else if (uri.match(/^tel:/i)) {
343 return "tel";
344 }
345 else if (uri.match(/^callto:/i)) {
346 return "callto";
347 }
348 else if (uri.match(/^telnet:/i)) {
349 return "telnet";
350 }
351 else if (uri.match(/^news:/i)) {
352 return "news";
353 }
354 else if (uri.match(/^nntp:/i)) {
355 return "nntp";
356 }
357 else if (uri.match(/^ftp:/i)) {
358 return "ftp";
359 }
360 else if (uri.match(/^mailto:/i)) {
361 return "email";
362 }
363 else if (uri.match(/^urn:/i)) {
364 return "urn";
365 }
366 else if (uri.match(/^data:/i)) {
367 return "data";
368 }
369 else if (uri.match(/^feed:/i)) {
370 return "feed";
371 }
372 return "unknown";
373}
374
375
376function guessServiceTypeByMime(mimeType)
377{
378 if (mimeType.match(/^application\/(rss|atom)\+xml/i)) {
379 dump("feed detected!\n");
380 return "feed";
381 }
382 else if (mimeType.match(/^image\//i)) {
383 return "image";
384 }
385 return null;
386}
387
388
389
390function friendlyServiceName(srvType, uri)
391{
392 if (srvType && srvType == "xri://+i-service*(+contact)*($v*1.0)") {
393 return "Contact Service";
394 }
395 else if (srvType && srvType == "http://openid.net/signon/1.0") {
396 return "OpenID Authentication Service";
397 }
398 else if (srvType && srvType == "xri://$res*auth*($v*2.0)") {
399 return "Authority Resolution Service";
400 }
401 else {
402 if (uri == null) {
403 return "Generic Service";
404 }
405 if (uri.match(/^https?:/i)) {
406 return "Web Link";
407 }
408 else if (uri.match(/^skype:/i)) {
409 var user = uri.substring("skype:".length, uri.indexOf('?'));
410 return "Skype <a href=\"" + uri + "\"><img src=\"chrome://foxri/content/skype_call.png\"></a>";
411 }
412 else if (uri.match(/^mailto:/i)) {
413 var qmark = uri.indexOf('?');
414 var email = (qmark == -1)?
415 uri.substr("mailto:".length) :
416 uri.substring("mailto:".length, qmark);
417 return "Email (address: " + email + ")";
418 }
419 else if (srvType != null) {
420 return srvType; // return verbatim
421 }
422 return "Generic Service";
423 }
424}
425
426
427
428
429function subHTML(template, vars)
430{
431 for (key in vars) {
432 template = template.replace(key, vars[key], 'g');
433 }
434 return template;
435}
436
437
438/// Given the completed XMLHttpRequest object, renders the XRDS
439function renderXRDS(xmlDoc)
440{
441 var x = xmlDoc;
442 var qxri = getString(x, x, "/xrds:XRDS/@ref");
443
444 var html = subHTML(HTML_HEAD, { '#QXRI#': qxri });
445
446 // TODO: render parents as well
447
448 var lastNode = getFinalXRD(x);
449 if (lastNode) {
450 var stat = getString(x, lastNode, "xrd:Status/@code");
451 if (stat == "100") {
452 html += "<h3>Exploring <strong>" + qxri + "</strong></h3>";
453 }
454 else {
455 var msg = getString(x, lastNode, "xrd:Status/text()");
456 html += "<h3 class=\"error\"><strong>" + qxri + "</strong> failed to resolve (reason: " + stat + " - " + msg + ")</h3>";
457 }
458
459 html += "<br/>";
460
461 var services = runExpr(x, lastNode, "xrd:Service");
462 var s;
463 var count = getNumeric(x, lastNode, "count(xrd:Service)");
464 if (count > 0) {
465 while (s = services.iterateNext()) {
466 count++;
467 html += renderService(s, x, qxri);
468 }
469 }
470 else if (stat == '222') {
471 var xriType = isIName(qxri)? 'I-name' : 'I-number';
472 html += "<p class='error'>" + xriType + " does not exist.</p>\n";
473 }
474 else {
475 html += "<p>No service has been configured for this XRI</p>";
476 }
477
478 }
479
480 html += "</html>";
481
482 return html;
483}
484
485
486
487
488
489
490/***********************************************************
491 XriServiceExplorer class definition
492 ***********************************************************/
493
494
495function XRIChannel(uri) {
496 this.URI = uri;
497 var r = uri.spec.indexOf('#');
498 if (r >= 0) {
499 this.qxri = uri.spec.substring(0, r);
500 this.fragment = uri.spec.substring(r);
501 }
502 else {
503 this.qxri = uri.spec;
504 }
505};
506
507
508XRIChannel.prototype = {
509
510 fragment: null,
511
512/* private fields used internally */
513 qxri: null,
514
515 xmlRequest: null,
516
517 renderedHTML: null,
518
519 scriptableInStream: null,
520
521 buf: null,
522
523 mChannel: null,
524
525
526 copyFields: function(request)
527 {
528 dump("copyFields(loadFlags=" + request.loadFlags + ")\n");
529 dump("loadGroup = " + request.loadGroup + "\n");
530 dump("notificationCallbacks = " + request.notificationCallbacks + "\n");
531
532 // copy request fields
533 this.loadFlags = request.loadFlags;
534 this.loadGroup = request.loadGroup;
535 this.name = request.name;
536 this.status = request.status;
537
538 var channel = request.QueryInterface(nsIChannel);
539 if (channel) {
540 this.contentCharset = channel.contentCharset;
541 this.contentLength = channel.contentLength;
542 this.contentType = channel.contentType; // XXX
543 this.contentType = "text/html";
544 this.notificationCallbacks = channel.notificationCallbacks;
545 this.originalURI = this.originalURI;
546 this.URI = this.URI;
547 this.owner = channel.owner;
548 this.securityInfo = channel.securityInfo;
549
550 channel = channel.QueryInterface(nsIHttpChannel);
551 if (channel) {
552 this.allowPipelining = channel.allowPipelining;
553 this.redirectionLimit = channel.redirectionLimit;
554 this.referrer = channel.referrer;
555 this.requestMethod = channel.requestMethod;
556 this.requestSucceeded = channel.requestSucceeded;
557 this.responseStatus = channel.responseStatus;
558 this.responseStatusText = channel.responseStatusText;
559 }
560 }
561
562 },
563
564 /* nsIStreamListener */
565 asyncOpenListener: null,
566
567 /* nsISupports (but we really don't care) */
568 asyncOpenContext: null,
569
570
571 /* has the XML finished loading? */
572 loadDone: false,
573
574
575
576/* public fields (nsIStreamListener implementation) */
577 onDataAvailable : function(request, ctx, inputStream, offset, count)
578 {
579 dump("\nonDataAvailable, offset=" + offset + ", count=" + count + "\n");
580
581 // XXX
582/*
583 this.copyFields(request);
584 this.asyncOpenListener.onDataAvailable(this, this.asyncOpenContext, inputStream, offset, count);
585 return;
586*/
587
588
589 if (offset == 0) {
590 this.scriptableInStream.init(inputStream);
591 }
592
593 this.buf += this.scriptableInStream.read(count);
594
595 if (!request.isPending()) {
596 dump("request finished, buf = " + this.buf + "\n");
597
598 this.scriptableInStream = null;
599 }
600 else {
601 dump("request pending...\n");
602 dump("buf so far = " + this.buf + "\n");
603 }
604 },
605
606
607/* public fields (nsIRequestObserver implementation) */
608 onStartRequest : function(request, ctx)
609 {
610 dump("\nonStartRequest called\n");
611 // XXX
612
613 this.copyFields(request);
614 this.asyncOpenListener.onStartRequest(this, this.asyncOpenContext);
615 },
616
617
618 onStopRequest : function(request, ctx, status)
619 {
620 dump("\nonStopRequest called - status " + status + "\n");
621
622 // XXX
623/*
624 this.asyncOpenListener.onStopRequest(this, this.asyncOpenContext, status);
625 return;
626*/
627
628 this.copyFields(request);
629 this.loadDone = true;
630
631 if (status == 0) {
632
633 var domParser = Components.classes["@mozilla.org/xmlextras/domparser;1"].createInstance(Components.interfaces.nsIDOMParser);
634 var xmlDoc = domParser.parseFromString(this.buf, "text/xml");
635
636 // make fake inputstream
637 var renderedHTML = renderXRDS(xmlDoc);
638
639 this.contentCharset = "UTF-8";
640 this.contentLength = renderedHTML.length;
641 this.contentType = "text/html";
642
643 dump("rendered HTML = \n" + renderedHTML + "\n");
644
645 dump("\nCalling asyncOpenListener.onStartRequest\n\n");
646
647
648 var strIStream = Components.classes["@mozilla.org/io/string-input-stream;1"].createInstance(Components.interfaces.nsIStringInputStream);
649 if (strIStream) {
650 strIStream.setData(renderedHTML, renderedHTML.length);
651/*
652 strIStream.setData(this.buf, this.buf.length);
653*/
654 dump("\nleftovers in string-input-stream = " + strIStream.available() + "\n");
655 dump("\nCalling asyncOpenListener.onDataAvailable\n\n");
656 this.asyncOpenListener.onDataAvailable(this, this.asyncOpenContext, strIStream, 0, renderedHTML.length);
657/*
658 this.asyncOpenListener.onDataAvailable(this, this.asyncOpenContext, strIStream, 0, this.buf.length);
659*/
660
661 dump("\nleftovers in string-input-stream = " + strIStream.available() + "\n");
662 }
663 }
664 else {
665 dump("\nStatus = " + status + "\n");
666 dump("Calling asyncOpenListener.onStartRequest\n\n");
667 this.asyncOpenListener.onStartRequest(this, this.asyncOpenContext);
668 }
669
670 dump("stopping request for underlying asyncOpenListener\n");
671
672 this.asyncOpenListener.onStopRequest(this, this.asyncOpenContext, status);
673
674 // copied from nsIWyciwygChannel
675 this.asyncOpenListener = null;
676 this.asyncOpenContext = null;
677
678/*
679 if (this.loadGroup) {
680 this.loadGroup.removeRequest(request, null, status);
681 }
682*/
683
684 this.notificationCallbacks = null;
685 this.mChannel = null;
686
687 dump("stopped request\n");
688 },
689
690
691/* public fields (nsIInputStream implementation) */
692 available: function()
693 {
694 dump("nsIInputStream::available called\n");
695 return renderedHTML.length;
696 },
697
698 close: function()
699 {
700 dump("nsIInputStream::close called\n");
701 },
702
703 isNonBlocking: function() {
704 dump("nsIInputStream::isNonBlocking called\n");
705 return true;
706 },
707
708 read: function() { dump("nsIInputStream::read() called!!!\n"); },
709
710
711
712
713
714/* public fields (nsIRequest implmentation) */
715
716 loadFlags: 0,
717
718 loadGroup: null,
719
720 name: "xri://request",
721
722 status: 0,
723
724 cancel: function(status) { dump("\ncancel called...\n"); },
725
726 isPending: function() {
727 dump("isPending called\n\n");
728 return !this.loadDone;
729 },
730
731 resume: function() { dump("resume called\n"); },
732
733 suspend: function() { dump("suspend called\n"); },
734
735
736
737/* public fields (nsIChannel implmentation) */
738
739 contentCharset: null,
740
741 contentLength: -1,
742
743 contentType: null,
744
745 notificationCallbacks: null,
746
747 originalURI: null,
748
749 owner: null,
750
751 securityInfo: null,
752
753 URI: null,
754
755 open: function()
756 {
757 dump("open not supporteD!!!!!!\n");
758 },
759
760 asyncOpen: function(listener, context)
761 {
762 dump("asyncOpen called!!!!!!\n");
763 this.asyncOpenListener = listener;
764 this.asyncOpenContext = context;
765
766 var hxri = PROXY_URI + this.qxri
767 + "?_xrd_r=application/xrds%2Bxml;sep=false";
768 var ioService = Components.classesByID[kIOSERVICE_CID_STR].getService();
769 ioService = ioService.QueryInterface(nsIIOService);
770 var channel = ioService.newChannel(hxri, null, null);
771
772 if (this.scriptableInStream) {
773 dump("Hey! You can't possibly be reusing this handler?!\n");
774 return;
775 }
776
777 dump("making scriptableInStream\n");
778 this.scriptableInStream = Components.classes["@mozilla.org/scriptableinputstream;1"]
779 .createInstance(Components.interfaces.nsIScriptableInputStream);
780
781 this.buf = '';
782
783 dump("notificationCallbacks = " + this.notificationCallbacks + "\n");
784 dump("loadFlags = " + this.loadFlags + "\n");
785 dump("loadGroup = " + this.loadGroup + "\n");
786 dump("owner = " + this.owner + "\n");
787 dump("securityInfo = " + this.securityInfo + "\n");
788
789 // these nsIRequest attributes must be copied to the stub
790 // channel that we created
791 channel.notificationCallbacks = this.notificationCallbacks;
792 channel.loadGroup = this.loadGroup;
793 channel.loadFlags = this.loadFlags;
794
795 this.mChannel = channel;
796 channel.asyncOpen(this, null);
797 },
798
799
800/* public fields (nsIChannel implmentation) */
801 allowPipelining: false,
802 redirectionLimit: 5,
803 referrer: "",
804 requestMethod: "GET",
805 requestSucceeded: true,
806 responseStatus: 200,
807 responseStatusText: "OK",
808 getRequestHeader: function(header) {
809 dump("getRequestHeader(" + header + ")\n");
810 var httpChannel = this.mChannel.QueryInterface(nsIHttpChannel);
811
812 try {
813 var val = httpChannel.getRequestHeader(header);
814 dump("getRequestHeader(" + header + ") = " + val + "\n");
815 return val;
816 }
817 catch (e) {
818 throw e;
819 }
820 },
821 getResponseHeader: function(header) {
822 dump("getResponseHeader(" + header + ")\n");
823 var httpChannel = this.mChannel.QueryInterface(nsIHttpChannel);
824
825 try {
826 var val = httpChannel.getResponseHeader(header);
827 dump("getResponseHeader(" + header + ") = " + val + "\n");
828 return val;
829 }
830 catch (e) {
831 throw e;
832 }
833/* XXX
834 if (header == "Content-Type")
835 return "text/html";
836*/
837 return null;
838 },
839 isNoCacheResponse: function() {
840 dump("isNoCacheResponse()\n");
841 var httpChannel = this.mChannel.QueryInterface(nsIHttpChannel);
842 return httpChannel.isNoCacheResponse();
843 },
844 isNoStoreResponse: function() {
845 dump("isNoStoreResponse()\n");
846 var httpChannel = this.mChannel.QueryInterface(nsIHttpChannel);
847 return httpChannel.isNoStoreResponse();
848 return true;
849 },
850 setRequestHeader: function(header, value, merge) {
851 dump("setRequestHeader(" + header + ", " + value + ")\n");
852 var httpChannel = this.mChannel.QueryInterface(nsIHttpChannel);
853 return httpChannel.setRequestHeader(header, value, merge);
854 },
855 setResponseHeader: function(header, value, merge) {
856 dump("setResponseHeader(" + header + ", " + value + ")\n");
857 var httpChannel = this.mChannel.QueryInterface(nsIHttpChannel);
858 return httpChannel.setResponseHeader(header, value, merge);
859 },
860 visitRequestHeaders: function(visitor) {
861 dump("visitRequestHeaders()\n");
862 },
863 visitResponseHeaders: function(visitor) {
864 dump("visitResponseHeaders()\n");
865 },
866
867 QueryInterface: function(iid)
868 {
869 dump("QI.. \n");
870 if (iid.equals(nsIChannel))
871 dump("QI(nsIChannel)\n");
872 else if (iid.equals(nsIHttpChannel))
873 dump("QI(nsIHttpChannel)\n");
874 else if (iid.equals(Components.interfaces.nsIUploadChannel))
875 dump("QI(nsIUploadChannel) - not supported\n");
876 else if (iid.equals(Components.interfaces.nsICachingChannel))
877 dump("QI(nsICachingChannel) - not supported\n");
878 else if (iid.equals(Components.interfaces.nsIClassInfo))
879 dump("QI(nsIClassInfo) - not supported\n");
880 else if (iid.equals(Components.interfaces.nsISecurityCheckedComponent))
881 dump("QI(nsISecurityCheckedComponent) - not supported\n");
882 else if (iid.equals(Components.interfaces.nsIWyciwygChannel))
883 dump("QI(nsIWyciwygChannel) - not supported\n");
884 else if (iid.equals(Components.interfaces.nsIMultiPartChannel))
885 dump("QI(nsIMultiPartChannel) - not supported\n");
886 else if (iid.equals(Components.interfaces.nsIHttpChannelInternal))
887 dump("QI(nsIHttpChannelInternal) - not supported\n");
888 else if (iid.equals(Components.interfaces.nsIWritablePropertyBag2))
889 dump("QI(nsIWritablePropertyBag2) - not supported\n");
890 else if (iid.equals(nsIRequest))
891 dump("QI(nsIRequest)\n");
892 else if (iid.equals(nsIRequestObserver))
893 dump("QI(nsIRequestObserver)\n");
894 else if (iid.equals(nsISupports))
895 dump("QI(nsISupports)\n");
896 else if (iid.equals(nsIStreamListener))
897 dump("QI(nsIStreamListener)\n");
898 else
899 dump("unknown " + iid + "\n");
900
901 if (iid.equals(nsISupports) ||
902 iid.equals(nsIRequest) ||
903 iid.equals(nsIRequestObserver) ||
904 iid.equals(nsIChannel) ||
905 iid.equals(nsIHttpChannel) ||
906 iid.equals(nsIStreamListener)
907 ) {
908 return this;
909 }
910
911 throw Components.results.NS_ERROR_NO_INTERFACE;
912 }
913};
914
915
916
917/***********************************************************
918 XriProtocolHandler class definition
919 ***********************************************************/
920
921//class constructor
922function XriProtocolHandler() {
923};
924
925// class definition
926XriProtocolHandler.prototype = {
927 defaultPort: 80, // HTTP
928
929 protocolFlags : nsIProtocolHandler.ALLOWS_PROXY | nsIProtocolHandler.ALLOWS_PROXY_HTTP,
930
931 scheme: "xri",
932
933 allowPort: function() {
934 return false; // only called for blacklisted ports, should respect
935 },
936
937 _newHttpChannel: function(aURI)
938 {
939 var HXRI = PROXY_URI + aURI.spec;
940 var ioService = Components.classesByID[kIOSERVICE_CID_STR].getService();
941 ioService = ioService.QueryInterface(nsIIOService);
942 var channel = ioService.newChannel(HXRI, null, null);
943 return channel;
944 },
945
946 newChannel: function(aURI)
947 {
948 // leave alone if path is not empty or just a single slash or query exists
949
950 dump("path='" + aURI.path + "'\n");
951 dump("query='" + aURI.query + "'\n");
952 dump("spec='" + aURI.spec + "'\n");
953
954 var slashPos = aURI.spec.indexOf('/', 'xri://'.length);
955 var qmarkPos = aURI.spec.indexOf('?');
956 dump("slashPos='" + slashPos + "'\n");
957 dump("qmarkPos='" + qmarkPos + "'\n");
958 if ((slashPos > 0 && slashPos < aURI.spec.length - 1) || qmarkPos > -1) {
959 return this._newHttpChannel(aURI);
960 }
961
962 var explorer = new XRIChannel(aURI);
963 return explorer;
964 },
965
966
967 newURI: function(spec, originCharset, baseURI)
968 {
969 var newSpec = spec;
970 if (baseURI != null) {
971 // standard-url (nsIURL) does not work with @-GCS
972 var baseURL = Components.classes[CID_URL].createInstance(nsIURL);
973 baseURL.spec = baseURI.spec;
974 newSpec = baseURL.resolve(spec);
975 }
976
977 var uri = Components.classes[CID_URI].createInstance(nsIURI);
978 uri.spec = newSpec;
979 return uri;
980 },
981
982 QueryInterface: function(aIID)
983 {
984 if (!aIID.equals(nsIProtocolHandler) &&
985 !aIID.equals(nsISupports))
986 throw Components.results.NS_ERROR_NO_INTERFACE;
987 return this;
988 }
989};
990
991
992/***********************************************************
993 class factory
994
995 This object is a member of the global-scope Components.classes.
996 It is keyed off of the contract ID. Eg:
997
998 myXriProtocolHandler = Components.classes["@dietrich.ganx4.com/helloworld;1"].
999 createInstance(Components.interfaces.nsIXriProtocolHandler);
1000
1001 ***********************************************************/
1002var XriProtocolHandlerFactory = {
1003 createInstance: function (aOuter, aIID)
1004 {
1005 if (aOuter != null)
1006 throw Components.results.NS_ERROR_NO_AGGREGATION;
1007 return (new XriProtocolHandler()).QueryInterface(aIID);
1008 }
1009};
1010
1011
1012/***********************************************************
1013 module definition (xpcom registration)
1014 ***********************************************************/
1015var XriProtocolHandlerModule = {
1016
1017 _firstTime: true,
1018
1019 registerSelf: function(aCompMgr, aFileSpec, aLocation, aType)
1020 {
1021 if (this._firstTime) {
1022 this._firstTime = false;
1023 throw Components.results.NS_ERROR_FACTORY_REGISTER_AGAIN;
1024 }
1025 aCompMgr = aCompMgr.
1026 QueryInterface(Components.interfaces.nsIComponentRegistrar);
1027 aCompMgr.registerFactoryLocation(CLASS_ID, CLASS_NAME,
1028 CONTRACT_ID, aFileSpec, aLocation, aType);
1029 },
1030
1031 unregisterSelf: function(aCompMgr, aLocation, aType)
1032 {
1033 aCompMgr = aCompMgr.
1034 QueryInterface(Components.interfaces.nsIComponentRegistrar);
1035 aCompMgr.unregisterFactoryLocation(CLASS_ID, aLocation);
1036 },
1037
1038 getClassObject: function(aCompMgr, aCID, aIID)
1039 {
1040 if (!aIID.equals(Components.interfaces.nsIFactory))
1041 throw Components.results.NS_ERROR_NOT_IMPLEMENTED;
1042
1043 if (aCID.equals(CLASS_ID))
1044 return XriProtocolHandlerFactory;
1045
1046 throw Components.results.NS_ERROR_NO_INTERFACE;
1047 },
1048
1049 canUnload: function(aCompMgr) { return true; }
1050};
1051
1052
1053/***********************************************************
1054 module initialization
1055
1056 When the application registers the component, this function
1057 is called.
1058 ***********************************************************/
1059function NSGetModule(aCompMgr, aFileSpec) { return XriProtocolHandlerModule; }
1060
diff --git a/src/install.rdf b/src/install.rdf
new file mode 100755
index 0000000..a482755
--- a/dev/null
+++ b/src/install.rdf
@@ -0,0 +1,40 @@
1<?xml version="1.0"?>
2
3<RDF xmlns="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
4 xmlns:em="http://www.mozilla.org/2004/em-rdf#">
5
6 <Description about="urn:mozilla:install-manifest">
7 <em:id>foxri@foxri.net</em:id>
8 <em:version>1.1.1</em:version>
9 <em:type>2</em:type>
10
11 <!-- Target Application this extension can install into,
12 with minimum and maximum supported versions. -->
13 <em:targetApplication>
14 <Description>
15 <em:id>{ec8030f7-c20a-464f-9b0e-13a3a9e97384}</em:id>
16 <em:minVersion>1.0</em:minVersion>
17 <em:maxVersion>1.5.0.*</em:maxVersion>
18 </Description>
19 </em:targetApplication>
20
21 <em:targetApplication>
22 <Description>
23 <em:id>{a463f10c-3994-11da-9945-000d60ca027b}</em:id>
24 <em:minVersion>0.5</em:minVersion>
25 <em:maxVersion>0.8</em:maxVersion>
26 </Description>
27 </em:targetApplication>
28
29
30 <!-- Front End MetaData -->
31 <em:name>FoXRI</em:name>
32 <em:description>XRI Extension</em:description>
33 <em:creator>William Tan</em:creator>
34 <em:homepageURL>http://dready.org/projects/foxri</em:homepageURL>
35<!--
36 <em:updateURL>http://dready.org/projects/foxri/update.rdf</em:updateURL>
37-->
38 </Description>
39</RDF>
40