e074a9a7b4ce5f950bc59a66f5daf2e91df6342c
[openx] / openx.js
1 /** Optimized methods for fetching ad-banners via OpenX */
2
3 /** see: http://enterprisejquery.com/2010/10/how-good-c-habits-can-encourage-bad-javascript-habits-part-1/ */
4
5 (function( openx, $, undefined ) {
6
7   var
8
9   id,
10   node,
11
12   count = 0,
13   slots = {},
14   queue = [],
15   ads = [],
16   output = [];
17
18
19   openx.show_ads = function(server, zones) {
20
21     var
22     domain = document.location.protocol == 'https:' ? 'https://' + server + ':8443':'http://' + server,
23     name,
24     src = domain;
25
26     /**
27      * Without this option, jQuery appends an timestamp to every URL, that
28      * is fetched via $.getScript(). This can mess up badly written
29      * third-party-ad-scripts, that assume that the called URL's are not
30      * altered.
31      */
32     $.ajaxSetup({ cache: true });
33
34     src += "/www/delivery/spc.php?zones=";
35
36     /** Only fetch banners, that are really included in this page */
37     for(name in zones) {
38       $('.oa').each(function() {
39         var
40         node = $(this),
41         id;
42         if (node.hasClass(name)) {
43           id = 'oa_' + ++count;
44           slots[id] = node;
45           queue.push(id);
46           src += escape(id + '=' + zones[name] + "|");
47         }
48       });
49     }
50
51     if (typeof OA_source !== 'undefined')
52       src += "&source=" + escape(OA_source);
53     src += "&nz=1&r=" + Math.floor(Math.random()*99999999);
54     src += "&block=1&charset=UTF-8";
55
56     if (window.location)   src += "&loc=" + escape(window.location);
57     if (document.referrer) src += "&referer=" + escape(document.referrer);
58
59     $.getScript(src, init_ads);
60
61     src = domain + '/www/delivery/fl.js';
62     $.getScript(src);
63
64   }
65
66   function init_ads() {
67
68     var i, id;
69     for (i=0; i<queue.length; i++) {
70       id = queue[i];
71       if (typeof(OA_output[id]) != 'undefined' && OA_output[id] != '')
72         ads.push(id);
73     }
74
75     document.write = document_write;
76     document.writeln = document_write;
77
78     render_ads();
79
80   }
81
82   function render_ads() {
83
84     while (ads.length > 0) {
85
86       var result, src, inline, i;
87
88       id = ads.shift();
89       node = slots[id];
90
91       node.slideDown();
92
93       // node.append(id + ": " + node.attr('class'));
94
95       /**
96        * If output was added via document.write(), this output must be
97        * rendered before other banner-code from the OpenX-server is rendered!
98        */
99       insert_output();
100
101       while ((result = /<script/i.exec(OA_output[id])) != null) {
102         node.append(OA_output[id].slice(0,result.index));
103         /** Strip all text before "<script" from OA_output[id] */
104         OA_output[id] = OA_output[id].slice(result.index,OA_output[id].length);
105         result = /^([^>]*)>([\s\S]*?)<\\?\/script>/i.exec(OA_output[id]);
106         if (result == null) {
107           /** Invalid syntax in delivered banner-code: ignoring the rest of this banner-code! */
108           // alert(OA_output[id]);
109           OA_output[id] = "";
110         }
111         else {
112           /** Remember iinline-code, if present */
113           src = result[1] + ' ' // << simplifies the following regular expression: the string ends with a space in any case, so that the src-URL cannot be followed by the end of the string emediately!
114           inline = result[2];
115           /** Strip all text up to and including "</script>" from OA_output[id] */
116           OA_output[id] = OA_output[id].slice(result[0].length,OA_output[id].length);
117           result = /src\s*=\s*['"]?([^'"]*)['"]?\s/i.exec(src);
118           if (result == null) {
119             /** script-tag with inline-code: execute inline-code! */
120             result = /^\s*<.*$/m.exec(inline);
121             if (result != null) {
122               /** Remove leading HTML-comments, because IE will stumble otherwise */
123               inline = inline.slice(result[0].length,inline.length);
124             }
125             $.globalEval(inline);
126             insert_output(); // << The executed inline-code might have called document.write()!
127           }
128           else {
129             /** script-tag with src-URL! */
130             if (OA_output[id].length > 0)
131               /** The banner-code was not rendered completely yet! */
132               ads.unshift(id);
133             /** Load the script and halt all work until the script is loaded and executed... */
134             $.getScript(result[1], render_ads); // << jQuery.getScript() generates onload-Handler for _all_ browsers ;)
135             return;
136           }
137         }
138       }
139
140       node.append(OA_output[id]);
141       OA_output[id] = "";
142     }
143
144     /** All entries from OA_output were rendered */
145
146     id = undefined;
147     node = undefined;
148   }
149
150   /** This function is used to overwrite document.write and document.writeln */
151   function document_write() {
152
153     if (id == undefined)
154       return;
155
156     for (var i=0; i<arguments.length; i++)
157       output.push(arguments[i]);
158
159     if (id != ads[0])
160       /**
161        * Re-Add the last banner-code to the working-queue, because included
162        * scripts had added markup via document.write(), which is not
163        * proccessed yet.
164        * Otherwise the added markup would be falsely rendered together with
165        * the markup from the following banner-code.
166        */
167       ads.unshift(id);
168
169   }
170
171   /**
172    * This function prepends the collected output from calls to
173    * document_write() to the current banner-code.
174    */
175   function insert_output() {
176
177     if (output.length > 0) {
178       output.push(OA_output[id]);
179       OA_output[id] = "";
180       for (i=0; i<output.length; i++)
181         OA_output[id] += output[i];
182       output = [];
183     }
184
185   }
186
187 } ( window.openx = window.openx || {}, jQuery ));
188
189 var OA_output = {}; // << Needed, because IE will complain loudly otherwise!