Record presentation.
[pazpar2-moved-to-github.git] / www / demo / search.js
1 /* $Id: search.js,v 1.6 2007-01-05 14:56:05 sondberg Exp $
2  * ---------------------------------------------------
3  * Javascript container
4  */
5
6 var xmlHttp
7 var xinitSession;
8 var xloadTargets;
9 var xsearch;
10 var xshow;
11 var xstat;
12 var xtermlist;
13 var session = false;
14 var targetsloaded = false;
15 var shown;
16 var searchtimer;
17 var showtimer;
18 var termtimer;
19 var stattimer;
20 var session_cells = Array('query', 'startrec', 'action_type');
21 var old_session = session_read();
22 var url_surveillence;
23 var recstoshow = 15;
24 var facet_list;
25 var cur_facet = 0;
26
27 function initialize ()
28 {
29     facet_list = get_available_facets();
30     start_session();
31     session_check();
32 }
33
34
35 function GetXmlHttpObject()
36
37     var objXMLHttp=null
38     if (window.XMLHttpRequest)
39       {
40       objXMLHttp=new XMLHttpRequest()
41       }
42     else if (window.ActiveXObject)
43       {
44       objXMLHttp=new ActiveXObject("Microsoft.XMLHTTP")
45       }
46     return objXMLHttp
47
48
49 function SendXmlHttpObject(obj, url, handler)
50 {
51     obj.onreadystatechange=handler;
52     obj.open("GET", url);
53     obj.send(null);
54 }
55
56 function session_started()
57 {
58     if (xinitSession.readyState != 4)
59         return;
60     var xml = xinitSession.responseXML;
61     var sesid = xml.getElementsByTagName("session")[0].childNodes[0].nodeValue;
62     document.getElementById("status").innerHTML = "Live";
63     session = sesid;
64     setTimeout(ping_session, 50000);
65 }
66
67 function start_session()
68 {
69     xinitSession = GetXmlHttpObject();
70     var url="search.pz2?";
71     url += "command=init";
72     xinitSession.onreadystatechange=session_started;
73     xinitSession.open("GET", url);
74     xinitSession.send(null);
75     
76     //url_surveillence = setInterval(session_check, 200);
77 }
78
79 function ping_session()
80 {
81     if (!session)
82         return;
83     var url = "search.pz2?command=ping&session=" + session;
84     SendXmlHttpObject(xpingSession = GetXmlHttpObject(), url, session_pinged);
85 }
86
87 function session_pinged()
88 {
89     if (xpingSession.readyState != 4)
90         return;
91     var xml = xpingSession.responseXML;
92     var error = xml.getElementsByTagName("error");
93     if (error[0])
94     {
95         var msg = error[0].childNodes[0].nodeValue;
96         alert(msg);
97         location = "?";
98         return;
99     }
100     setTimeout(ping_session, 50000);
101 }
102
103 function targets_loaded()
104 {
105     if (xloadTargets.readyState != 4)
106         return;
107     var xml = xloadTargets.responseXML;
108     var error = xml.getElementsByTagName("error");
109     if (error[0])
110     {
111         var msg = error[0].childNodes[0].nodeValue;
112         alert(msg);
113         return;
114     }
115     document.getElementById("targetstatus").innerHTML = "Targets loaded";
116 }
117
118 function load_targets()
119 {
120     var fn = document.getElementById("targetfilename").value;
121     clearTimeout(termtimer);
122     clearTimeout(searchtimer);
123     clearTimeout(stattimer);
124     clearTimeout(showtimer);
125     document.getElementById("stat").innerHTML = "";
126     if (!fn)
127     {
128         alert("Please enter a target definition file name");
129         return;
130     }
131     var url="search.pz2?" +
132         "command=load" +
133         "&session=" + session +
134         "&name=" + fn;
135     document.getElementById("targetstatus").innerHTML = "Loading targets...";
136     xloadTargets = GetXmlHttpObject();
137     xloadTargets.onreadystatechange=targets_loaded;
138     xloadTargets.open("GET", url);
139     xloadTargets.send(null);
140 }
141
142
143 function update_action (new_action) {
144     document.search.action_type.value = new_action;
145 }
146
147
148 function make_pager (hits, offset, max) {
149     return '<a href="#" class="select">1</a> <a href="#">Next</a>';
150 }
151
152
153 function show_records()
154 {
155     if (xshow.readyState != 4)
156         return;
157     var i;
158     var xml = xshow.responseXML;
159     var body = document.getElementById("body");
160     var hits = xml.getElementsByTagName("hit");
161     if (!hits[0]) // We should never get here with blocking operations
162     {
163         body.innerHTML = "No records yet";
164         searchtimer = setTimeout(check_search, 250);
165     }
166     else
167     {
168
169         var total = Number(xml.getElementsByTagName('total')[0].childNodes[0].nodeValue);
170         var merged = Number(xml.getElementsByTagName('merged')[0].childNodes[0].nodeValue);
171         var start = Number(xml.getElementsByTagName('start')[0].childNodes[0].nodeValue);
172         var num = Number(xml.getElementsByTagName('num')[0].childNodes[0].nodeValue);
173         var clients = Number(xml.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue);
174         body.innerHTML = '<div class="pages">' +
175                          make_pager(merged, start, 20) +
176                          '</div>';
177                          
178         body.innerHTML += '<div class="results">Records : ' + (start + 1) +
179                           ' to ' + (start + num) + ' of ' + merged +
180                           ' (total hits: ' + total + ')</div><br/><br/>';
181
182 /*
183         if (start + num < merged)
184             body.innerHTML += ' <a href="" ' +
185                 'onclick="document.search.startrec.value=' + (start + recstoshow) +
186                 ";update_action('page')" +
187                 ';check_search(); update_history(); return false;">Next</a>';
188
189         if (start > 0)
190             body.innerHTML += ' <a href="" ' +
191                 'onclick="document.search.startrec.value=' + (start - recstoshow) +
192                 ";update_action('page')" +
193                 ';check_search(); update_history();return false;">Previous</a>';
194
195         body.innerHTML += '<br/>';
196 */
197         body.innerHTML += '<div class="records">';
198
199         for (i = 0; i < hits.length; i++)
200         {
201             var mk = hits[i].getElementsByTagName("title");
202
203             body.innerHTML += '<a href="#" class="record">';
204
205             if (mk[0]) {
206                 var field = mk[0].childNodes[0].nodeValue;
207             }
208             body.innerHTML += field + '</a>';
209         }
210
211         body.innerHTML += '</div>';
212         shown++;
213         if (clients > 0)
214         {
215             if (shown < 5)
216                 searchtimer = setTimeout(check_search, 1000);
217             else
218                 searchtimer = setTimeout(check_search, 2000);
219         }
220     }
221     if (!termtimer)
222         termtimer = setTimeout(check_termlist, 500);
223 }
224
225 function check_search()
226 {
227     clearTimeout(searchtimer);
228     var url = "search.pz2?" +
229         "command=show" +
230         "&start=" + document.search.startrec.value +
231         "&num=" + recstoshow +
232         "&session=" + session +
233         "&block=1";
234     xshow = GetXmlHttpObject();
235     xshow.onreadystatechange=show_records;
236     xshow.open("GET", url);
237     xshow.send(null);
238 }
239
240
241 function refine_query (obj) {
242     var query_cell = document.getElementById('query');
243     var term = obj.innerHTML;
244     
245     term = term.replace(/[\(\)]/g, '');
246     if (cur_termlist == 'subject')
247         query_cell.value += ' and su=(' + term + ')';
248     else if (cur_termlist == 'author')
249         query_cell.value += ' and au=(' + term + ')';
250     start_search();
251 }
252
253
254
255 function show_termlist()
256 {
257     if (xtermlist.readyState != 4)
258         return;
259
260     var i;
261     var xml = xtermlist.responseXML;
262     var body = facet_list[cur_facet][1];
263     var hits = xml.getElementsByTagName("term");
264     var clients =
265         Number(xml.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue);
266
267     cur_facet++;
268
269     if (cur_facet >= facet_list.length)
270         cur_facet = 0;
271
272     if (!hits[0])
273     {
274         termtimer = setTimeout(check_termlist, 500);
275     }
276     else
277     {
278         body.innerHTML = '';
279         
280         for (i = 0; i < hits.length; i++)
281         {
282             var namen = hits[i].getElementsByTagName("name");
283             if (namen[0])
284                 body.innerHTML += '<a href="#" onclick="refine_query(this)">' +
285                                   namen[0].childNodes[0].nodeValue +
286                                   '</a>';
287         }
288
289         if (clients > 0)
290             termtimer = setTimeout(check_termlist, 1000);
291     }
292 }
293
294 function check_termlist()
295 {
296     var facet_name = facet_list[cur_facet][0];
297     var url = "search.pz2?" +
298         "command=termlist" +
299         "&session=" + session +
300         "&name=" + facet_name;
301     xtermlist = GetXmlHttpObject();
302     xtermlist.onreadystatechange=show_termlist;
303     xtermlist.open("GET", url);
304     xtermlist.send(null);
305 }
306
307 function show_stat()
308 {
309     if (xstat.readyState != 4)
310         return;
311     var i;
312     var xml = xstat.responseXML;
313     var body = document.getElementById("stat");
314     var nodes = xml.childNodes[0].childNodes;
315     var clients =
316         Number(xml.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue);
317     if (!nodes[0])
318     {
319         stattimer  = setTimeout(check_stat, 500);
320     }
321     else
322     {
323         body.innerHTML = "(";
324         for (i = 0; i < nodes.length; i++)
325         {
326             if (nodes[i].nodeType != 1)
327                 continue;
328             var value = nodes[i].childNodes[0].nodeValue;
329             if (value == 0)
330                 continue;
331             var name = nodes[i].nodeName;
332             body.innerHTML += ' ' + name + '=' + value;
333         }
334         body.innerHTML += ')';
335         if (clients > 0)
336             stattimer = setTimeout(check_stat, 2000);
337     }
338 }
339
340 function check_stat()
341 {
342     var url = "search.pz2?" +
343         "command=stat" +
344         "&session=" + session;
345     xstat = GetXmlHttpObject();
346     xstat.onreadystatechange=show_stat;
347     xstat.open("GET", url);
348     xstat.send(null);
349 }
350
351 function search_started()
352 {
353     if (xsearch.readyState != 4)
354         return;
355     var xml = xsearch.responseXML;
356     var error = xml.getElementsByTagName("error");
357     if (error[0])
358     {
359         var msg = error[0].childNodes[0].nodeValue;
360         alert(msg);
361         return;
362     }
363     check_search();
364     stattimer = setTimeout(check_stat, 1000);
365 }
366
367 function start_search()
368 {
369     clearTimeout(termtimer);
370     termtimer = 0;
371     clearTimeout(searchtimer);
372     searchtimer = 0;
373     clearTimeout(stattimer);
374     stattimer = 0;
375     clearTimeout(showtimer);
376     showtimer = 0;
377     if (!targets_loaded)
378     {
379         alert("Please load targets first");
380         return;
381     }
382     var query = escape(document.getElementById('query').value);
383     var url = "search.pz2?" +
384         "command=search" +
385         "&session=" + session +
386         "&query=" + query;
387     xsearch = GetXmlHttpObject();
388     xsearch.onreadystatechange=search_started;
389     xsearch.open("GET", url);
390     xsearch.send(null);
391 //    document.getElementById("termlist").innerHTML = '';
392     document.getElementById("body").innerHTML = '';
393     update_history();
394     shown = 0;
395     document.search.startrec.value = 0;
396 }
397
398
399 function session_encode ()
400 {
401     var i;
402     var session = '';
403
404     for (i = 0; i < session_cells.length; i++)
405     {
406         var name = session_cells[i];
407         var value = escape(document.getElementById(name).value);
408         session += '&' + name + '=' + value;
409     }
410
411     return session;
412 }
413
414
415 function session_restore (session)
416 {
417     var fields = session.split(/&/);
418     var i;
419
420     for (i = 1; i < fields.length; i++)
421     {
422         var pair = fields[i].split(/=/);
423         var key = pair.shift();
424         var value = pair.join('=');
425         var cell = document.getElementById(key);
426
427         cell.value = value;
428     }
429     
430 }
431
432
433 function session_read ()
434 {
435     var ses = window.location.hash.replace(/^#/, '');
436     return ses;
437 }
438
439
440 function session_store (new_value)
441 {
442     window.location.hash = '#' + new_value;
443 }
444
445
446 function update_history ()
447 {
448     var session = session_encode();
449     session_store(session);
450     old_session = session;
451 }
452
453
454 function session_check ()
455 {
456     var session = session_read();
457     var action = document.search.action_type.value;
458
459     clearInterval(url_surveillence);
460
461     if ( session != unescape(old_session) )
462     {
463         session_restore(session);
464
465         if (action == 'search') {
466             start_search();
467         } else if (action == 'page') {
468             check_search();
469         } else {
470             alert('Unregocnized action_type: ' + action);
471             return;
472         }
473     }
474     
475     url_surveillence = setInterval(session_check, 200);
476 }
477
478
479 function get_available_facets () {
480     var facet_container = document.getElementById('termlists');
481     var facet_cells = facet_container.childNodes;
482     var facets = Array();
483     var i;
484
485     for (i = 0; i < facet_cells.length; i++) {
486         var cell = facet_cells.item(i);
487
488         if (cell.className == 'facet') {
489             var facet_name = cell.id.replace(/^facet_([^_]+)_terms$/, "$1");
490             facets.push(Array(facet_name, cell));
491         }
492     }
493
494     return facets;
495 }
496
497
498 function get_facet_container (obj) {
499     return document.getElementById(obj.id + '_terms');
500 }
501
502
503 function toggle_facet (obj) {
504     var container = get_facet_container(obj);
505
506     if (obj.className == 'selected') {
507         obj.className = 'unselected';
508         container.style.display = 'inline';
509     } else {
510         obj.className = 'selected';
511         container.style.display = 'none';
512     }
513 }