2 ** $Id: pz2.js,v 1.30 2007-06-04 12:24:59 jakub Exp $
3 ** pz2.js - pazpar2's javascript client library.
6 //since explorer is flawed
8 window.Node = new Object();
10 Node.ATTRIBUTE_NODE = 2;
12 Node.CDATA_SECTION_NODE = 4;
13 Node.ENTITY_REFERENCE_NODE = 5;
15 Node.PROCESSING_INSTRUCTION_NODE = 7;
16 Node.COMMENT_NODE = 8;
17 Node.DOCUMENT_NODE = 9;
18 Node.DOCUMENT_TYPE_NODE = 10;
19 Node.DOCUMENT_FRAGMENT_NODE = 11;
20 Node.NOTATION_NODE = 12;
23 // prevent execution of more than once
24 if(typeof window.pz2 == "undefined") {
25 window.undefined = window.undefined;
27 var pz2 = function(paramArray) {
31 //supported pazpar2's protocol version
32 __myself.suppProtoVer = '1';
33 __myself.pz2String = "search.pz2";
34 __myself.stylesheet = paramArray.detailstylesheet || null;
36 //load stylesheet if required in async mode
37 if( __myself.stylesheet ) {
38 var request = new pzHttpRequest( __myself.stylesheet );
42 __myself.xslDoc = doc;
47 // at least one callback required
49 throw new Error("An array with parameters has to be suplied when instantiating a class");
51 __myself.errorHandler = paramArray.errorhandler || null;
54 __myself.statCallback = paramArray.onstat || null;
55 __myself.showCallback = paramArray.onshow || null;
56 __myself.termlistCallback = paramArray.onterm || null;
57 __myself.recordCallback = paramArray.onrecord || null;
58 __myself.bytargetCallback = paramArray.onbytarget || null;
59 __myself.resetCallback = paramArray.onreset || null;
62 __myself.termKeys = paramArray.termlist || "subject";
64 // some configurational stuff
65 __myself.keepAlive = 50000;
67 __myself.sessionID = null;
68 __myself.initStatusOK = false;
69 __myself.pingStatusOK = false;
70 __myself.searchStatusOK = false;
72 if ( paramArray.keepAlive < __myself.keepAlive )
73 __myself.keepAlive = paramArray.keepAlive;
76 __myself.currentSort = "relevance";
78 __myself.currentStart = 0;
79 __myself.currentNum = 20;
81 // last full record retrieved
82 __myself.currRecID = null;
84 __myself.currQuery = null;
87 __myself.statTime = paramArray.stattime || 1000;
88 __myself.statTimer = null;
89 __myself.termTime = paramArray.termtime || 1000;
90 __myself.termTimer = null;
91 __myself.showTime = paramArray.showtime || 1000;
92 __myself.showTimer = null;
93 __myself.showFastCount = 4;
94 __myself.bytargetTime = paramArray.bytargettime || 1000;
95 __myself.bytargetTimer = null;
97 // counters for each command and applied delay
98 __myself.dumpFactor = 500;
99 __myself.showCounter = 0;
100 __myself.termCounter = 0;
101 __myself.statCounter = 0;
102 __myself.bytargetCounter = 0;
104 // active clients, updated by stat and show
105 // might be an issue since bytarget will poll accordingly
106 __myself.activeClients = 1;
108 // auto init session?
109 if (paramArray.autoInit !== false)
115 __myself.sessionID = null;
116 __myself.initStatusOK = false;
117 __myself.pingStatusOK = false;
118 __myself.searchStatusOK = false;
120 clearTimeout(__myself.statTimer);
121 clearTimeout(__myself.showTimer);
122 clearTimeout(__myself.termTimer);
123 clearTimeout(__myself.bytargetTimer);
125 if ( __myself.resetCallback )
126 __myself.resetCallback();
128 init: function ( sessionId )
131 if ( sessionId != undefined ) {
132 __myself.initStatusOK = true;
133 __myself.sessionID = sessionId;
137 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
139 { "command": "init" },
141 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
142 if ( data.getElementsByTagName("protocol")[0].childNodes[0].nodeValue != __myself.suppProtoVer )
143 throw new Error("Server's protocol not supported by the client");
144 __myself.initStatusOK = true;
145 __myself.sessionID = data.getElementsByTagName("session")[0].childNodes[0].nodeValue;
146 setTimeout("__myself.ping()", __myself.keepAlive);
149 // if it gets here the http return code was 200 (pz2 errors are 417)
150 // but the response was invalid, it should never occur
151 setTimeout("__myself.init()", 1000);
156 // no need to ping explicitly
159 if( !__myself.initStatusOK )
161 // session is not initialized code here
162 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
164 { "command": "ping", "session": __myself.sessionID },
166 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
167 __myself.pingStatusOK = true;
168 setTimeout("__myself.ping()", __myself.keepAlive);
171 // if it gets here the http return code was 200 (pz2 errors are 417)
172 // but the response was invalid, it should never occur
173 setTimeout("__myself.ping()", 1000);
177 search: function (query, num, sort, filter)
179 clearTimeout(__myself.statTimer);
180 clearTimeout(__myself.showTimer);
181 clearTimeout(__myself.termTimer);
182 clearTimeout(__myself.bytargetTimer);
184 __myself.showCounter = 0;
185 __myself.termCounter = 0;
186 __myself.bytargetCounter = 0;
187 __myself.statCounter = 0;
189 if( !__myself.initStatusOK )
192 if( query !== undefined )
193 __myself.currQuery = query;
195 throw new Error("You need to supply query to the search command");
197 if( filter !== undefined )
198 var searchParams = { "command": "search", "session": __myself.sessionID, "query": __myself.currQuery, "filter": filter };
200 var searchParams = { "command": "search", "session": __myself.sessionID, "query": __myself.currQuery };
201 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
205 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
206 __myself.searchStatusOK = true;
208 __myself.show(0, num, sort);
209 if ( __myself.statCallback )
211 //__myself.statTimer = setTimeout("__myself.stat()", __myself.statTime / 4);
212 if ( __myself.termlistCallback )
214 //__myself.termTimer = setTimeout("__myself.termlist()", __myself.termTime / 4);
215 if ( __myself.bytargetCallback )
217 //__myself.bytargetTimer = setTimeout("__myself.bytarget()", __myself.bytargetTime / 4);
220 // if it gets here the http return code was 200 (pz2 errors are 417)
221 // but the response was invalid, it should never occur
222 setTimeout("__myself.search(__myself.currQuery)", 500);
228 if( !__myself.initStatusOK )
230 // if called explicitly takes precedence
231 clearTimeout(__myself.statTimer);
232 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
234 { "command": "stat", "session": __myself.sessionID },
236 if ( data.getElementsByTagName("stat") ) {
237 var activeClients = Number( data.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue );
238 __myself.activeClients = activeClients;
240 "activeclients": activeClients,
241 "hits": Number( data.getElementsByTagName("hits")[0].childNodes[0].nodeValue ),
242 "records": Number( data.getElementsByTagName("records")[0].childNodes[0].nodeValue ),
243 "clients": Number( data.getElementsByTagName("clients")[0].childNodes[0].nodeValue ),
244 "initializing": Number( data.getElementsByTagName("initializing")[0].childNodes[0].nodeValue ),
245 "searching": Number( data.getElementsByTagName("searching")[0].childNodes[0].nodeValue ),
246 "presenting": Number( data.getElementsByTagName("presenting")[0].childNodes[0].nodeValue ),
247 "idle": Number( data.getElementsByTagName("idle")[0].childNodes[0].nodeValue ),
248 "failed": Number( data.getElementsByTagName("failed")[0].childNodes[0].nodeValue ),
249 "error": Number( data.getElementsByTagName("error")[0].childNodes[0].nodeValue )
252 __myself.statCounter++;
253 var delay = __myself.statTime + __myself.statCounter * __myself.dumpFactor;
254 if ( activeClients > 0 )
255 __myself.statTimer = setTimeout("__myself.stat()", delay);
257 __myself.statCallback(stat);
260 // if it gets here the http return code was 200 (pz2 errors are 417)
261 // but the response was invalid, it should never occur
262 __myself.statTimer = setTimeout("__myself.stat()", __myself.statTime / 4);
266 show: function(start, num, sort)
268 if( !__myself.searchStatusOK )
270 // if called explicitly takes precedence
271 clearTimeout(__myself.showTimer);
273 if( sort !== undefined )
274 __myself.currentSort = sort;
275 if( start !== undefined )
276 __myself.currentStart = Number( start );
277 if( num !== undefined )
278 __myself.currentNum = Number( num );
279 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
282 { "command": "show", "session": __myself.sessionID, "start": __myself.currentStart,
283 "num": __myself.currentNum, "sort": __myself.currentSort, "block": 1 },
285 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
286 // first parse the status data send along with records
287 // this is strictly bound to the format
288 var activeClients = Number( data.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue );
289 __myself.activeClients = activeClients;
291 "activeclients": activeClients,
292 "merged": Number( data.getElementsByTagName("merged")[0].childNodes[0].nodeValue ),
293 "total": Number( data.getElementsByTagName("total")[0].childNodes[0].nodeValue ),
294 "start": Number( data.getElementsByTagName("start")[0].childNodes[0].nodeValue ),
295 "num": Number( data.getElementsByTagName("num")[0].childNodes[0].nodeValue ),
298 // parse all the first-level nodes for all <hit> tags
299 var hits = data.getElementsByTagName("hit");
300 var hit = new Array();
301 for (i = 0; i < hits.length; i++) {
302 show.hits[i] = new Array();
303 show.hits[i]['location'] = new Array();
304 for ( j = 0; j < hits[i].childNodes.length; j++) {
306 if ( hits[i].childNodes[j].nodeType == Node.ELEMENT_NODE ) {
307 if (hits[i].childNodes[j].nodeName == 'location') {
308 var locNode = hits[i].childNodes[j];
309 var id = locNode.getAttribute('id');
310 show.hits[i]['location'][id] = {
311 "id": locNode.getAttribute("id"),
312 "name": locNode.getAttribute("name")
316 var nodeName = hits[i].childNodes[j].nodeName;
317 var nodeText = hits[i].childNodes[j].firstChild.nodeValue;
318 show.hits[i][nodeName] = nodeText;
323 __myself.showCounter++;
324 var delay = __myself.showTime;
325 if (__myself.showCounter > __myself.showFastCount)
326 delay += __myself.showCounter * __myself.dumpFactor;
327 if ( activeClients > 0 )
328 __myself.showTimer = setTimeout("__myself.show()", delay);
330 __myself.showCallback(show);
333 // if it gets here the http return code was 200 (pz2 errors are 417)
334 // but the response was invalid, it should never occur
335 __myself.showTimer = setTimeout("__myself.show()", __myself.showTime / 4);
341 if( !__myself.searchStatusOK )
344 if( id !== undefined )
345 __myself.currRecID = id;
346 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
348 { "command": "record", "session": __myself.sessionID, "id": __myself.currRecID },
351 var record = new Array();
352 if ( recordNode = data.getElementsByTagName("record")[0] ) {
353 // if stylesheet was fetched do not parse the response
354 if ( __myself.xslDoc ) {
355 record['recid'] = recordNode.getElementsByTagName("recid")[0].firstChild.nodeValue;
356 record['xmlDoc'] = data;
357 record['xslDoc'] = __myself.xslDoc;
359 for ( i = 0; i < recordNode.childNodes.length; i++) {
360 if ( recordNode.childNodes[i].nodeType == Node.ELEMENT_NODE
361 && recordNode.childNodes[i].nodeName != 'location' ) {
362 var nodeName = recordNode.childNodes[i].nodeName;
363 var nodeText = recordNode.childNodes[i].firstChild.nodeValue;
364 record[nodeName] = nodeText;
367 // the location might be empty!!
368 var locationNodes = recordNode.getElementsByTagName("location");
369 record["location"] = new Array();
370 for ( i = 0; i < locationNodes.length; i++ ) {
371 record["location"][i] = {
372 "id": locationNodes[i].getAttribute("id"),
373 "name": locationNodes[i].getAttribute("name")
376 for ( j = 0; j < locationNodes[i].childNodes.length; j++) {
377 if ( locationNodes[i].childNodes[j].nodeType == Node.ELEMENT_NODE ) {
378 var nodeName = locationNodes[i].childNodes[j].nodeName;
380 if (locationNodes[i].childNodes[j].firstChild)
381 nodeText = locationNodes[i].childNodes[j].firstChild.nodeValue;
382 record["location"][i][nodeName] = nodeText;
388 __myself.recordCallback(record);
391 // if it gets here the http return code was 200 (pz2 errors are 417)
392 // but the response was invalid, it should never occur
393 setTimeout("__myself.record(__myself.currRecID)", 500);
399 if( !__myself.searchStatusOK )
401 // if called explicitly takes precedence
402 clearTimeout(__myself.termTimer);
403 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
405 { "command": "termlist", "session": __myself.sessionID, "name": __myself.termKeys },
407 if ( data.getElementsByTagName("termlist") ) {
408 var activeClients = Number( data.getElementsByTagName("activeclients")[0].childNodes[0].nodeValue );
409 __myself.activeClients = activeClients;
410 var termList = { "activeclients": activeClients };
411 var termLists = data.getElementsByTagName("list");
413 for (i = 0; i < termLists.length; i++) {
414 var listName = termLists[i].getAttribute('name');
415 termList[listName] = new Array();
416 var terms = termLists[i].getElementsByTagName('term');
417 //for each term in the list
418 for (j = 0; j < terms.length; j++) {
420 "name": terms[j].getElementsByTagName("name")[0].childNodes[0].nodeValue || 'ERROR',
421 "freq": terms[j].getElementsByTagName("frequency")[0].childNodes[0].nodeValue || 'ERROR'
424 var termIdNode = terms[j].getElementsByTagName("id");
425 if(terms[j].getElementsByTagName("id").length)
426 term["id"] = termIdNode[0].childNodes[0].nodeValue;
428 termList[listName][j] = term;
432 __myself.termCounter++;
433 var delay = __myself.termTime + __myself.termCounter * __myself.dumpFactor;
434 if ( activeClients > 0 )
435 __myself.termTimer = setTimeout("__myself.termlist()", delay);
437 __myself.termlistCallback(termList);
440 // if it gets here the http return code was 200 (pz2 errors are 417)
441 // but the response was invalid, it should never occur
442 __myself.termTimer = setTimeout("__myself.termlist()", __myself.termTime / 4);
449 if( !__myself.searchStatusOK )
451 // if called explicitly takes precedence
452 clearTimeout(__myself.bytargetTimer);
453 var request = new pzHttpRequest(__myself.pz2String, __myself.errorHandler);
455 { "command": "bytarget", "session": __myself.sessionID },
457 if ( data.getElementsByTagName("status")[0].childNodes[0].nodeValue == "OK" ) {
458 var targetNodes = data.getElementsByTagName("target");
459 var bytarget = new Array();
460 for ( i = 0; i < targetNodes.length; i++) {
461 bytarget[i] = new Array();
462 for( j = 0; j < targetNodes[i].childNodes.length; j++ ) {
463 if ( targetNodes[i].childNodes[j].nodeType == Node.ELEMENT_NODE ) {
464 var nodeName = targetNodes[i].childNodes[j].nodeName;
465 var nodeText = targetNodes[i].childNodes[j].firstChild.nodeValue;
466 bytarget[i][nodeName] = nodeText;
471 __myself.bytargetCounter++;
472 var delay = __myself.bytargetTime + __myself.bytargetCounter * __myself.dumpFactor;
473 if ( __myself.activeClients > 0 )
474 __myself.bytargetTimer = setTimeout("__myself.bytarget()", delay);
476 __myself.bytargetCallback(bytarget);
479 // if it gets here the http return code was 200 (pz2 errors are 417)
480 // but the response was invalid, it should never occur
481 __myself.bytargetTimer = setTimeout("__myself.bytarget()", __myself.bytargetTime / 4);
485 // just for testing, probably shouldn't be here
486 showNext: function(page)
488 var step = page || 1;
489 __myself.show( ( step * __myself.currentNum ) + __myself.currentStart );
491 showPrev: function(page)
493 if (__myself.currentStart == 0 )
495 var step = page || 1;
496 var newStart = __myself.currentStart - (step * __myself.currentNum );
497 __myself.show( newStart > 0 ? newStart : 0 );
499 showPage: function(pageNum)
501 //var page = pageNum || 1;
502 __myself.show(pageNum * __myself.currentNum);
507 *********************************************************************************
508 ** AJAX HELPER CLASS ************************************************************
509 *********************************************************************************
511 var pzHttpRequest = function ( url, errorHandler ) {
514 this.errorHandler = errorHandler || null;
516 if ( window.XMLHttpRequest ) {
517 this.request = new XMLHttpRequest();
518 } else if ( window.ActiveXObject ) {
520 this.request = new ActiveXObject( 'Msxml2.XMLHTTP' );
522 this.request = new ActiveXObject( 'Microsoft.XMLHTTP' );
527 pzHttpRequest.prototype =
529 get: function ( params, callback )
531 this.callback = callback;
533 var getUrl = this.url;
534 var paramArr = new Array();
536 for ( var key in params ) {
537 paramArr.push(key + '=' + encodeURI(params[key]) );
540 if ( paramArr.length )
541 getUrl += '?' + paramArr.join('&');
544 this.request.open( 'GET', getUrl, true );
545 this.request.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8');
546 //this.request.setRequestHeader('Accept-Charset', 'UTF-8');
547 this.request.onreadystatechange = function () {
548 context._handleResponse();
550 this.request.send(null);
553 _handleResponse: function ()
555 if ( this.request.readyState == 4 ) {
556 if ( this.request.status == 200 ) {
557 this.callback( this.request.responseXML );
560 else if ( this.request.status == 417 ) {
561 var errMsg = this.request.responseXML.getElementsByTagName("error")[0].childNodes[0].nodeValue;
562 var errCode = this.request.responseXML.getElementsByTagName("error")[0].getAttribute("code");
564 var err = new Error(errMsg);
567 if (this.errorHandler) {
568 this.errorHandler(err);
575 var err = new Error("XMLHttpRequest error. STATUS: "
576 + this.request.status + " STATUS TEXT: "
577 + this.request.statusText );
580 if (this.errorHandler) {
581 this.errorHandler(err);
592 *********************************************************************************
593 ** QUERY CLASS ******************************************************************
594 *********************************************************************************
596 var pzQuery = function()
598 this.simpleQuery = '';
599 this.singleFilter = null;
600 this.advTerms = new Array();
601 this.filterHash = new Array();
605 pzQuery.prototype = {
608 this.simpleQuery = '';
609 this.advTerms = new Array();
610 this.simpleFilter = null;
613 addTerm: function(field, value)
615 var term = {"field": field, "value": value};
616 this.advTerms[this.numTerms] = term;
619 getTermValueByIdx: function(index)
621 return this.advTerms[index].value;
623 getTermFieldByIdx: function(index)
625 return this.advTerms[index].field;
627 /* semicolon separated list of terms for given field*/
628 getTermsByField: function(field)
631 for(var i = 0; i < this.advTerms.length; i++)
633 if( this.advTerms[i].field == field )
634 terms = terms + this.queryHas[i].value + ';';
638 addTermsFromList: function(inputString, field)
640 var inputArr = inputString.split(';');
641 for(var i=0; i < inputArr.length; i++)
643 if(inputArr[i].length < 3) continue;
644 this.advTerms[this.numTerms] = {"field": field, "value": inputArr[i] };
648 removeTermByIdx: function(index)
650 this.advTerms.splice(index, 1);
656 if( this.simpleQuery != '')
657 ccl = this.simpleQuery;
658 for(var i = 0; i < this.advTerms.length; i++)
660 if (ccl != '') ccl = ccl + ' and ';
661 ccl = ccl + this.advTerms[i].field+'="'+this.advTerms[i].value+'"';
665 addFilter: function(name, value)
667 var filter = {"name": name, "id": value };
668 this.filterHash[this.filterHash.length] = filter;
670 return this.filterHash.length - 1;
672 setFilter: function(name, value)
674 this.filterHash = new Array();
676 this.addFilter(name, value);
678 getFilter: function(index)
680 return this.filterHash[index].id;
682 getFilterName: function(index)
684 return this.filterHash[index].name;
686 removeFilter: function(index)
688 delete this.filterHash[index];
691 clearFilter: function()
693 this.filterHash = new Array();
696 getFilterString: function()
699 if( this.singleFilter != null ) {
700 return 'pz:id='+this.singleFilter.id;
702 else if( this.filterNums <= 0 ) {
706 var filter = 'pz:id=';
707 for(var i = 0; i < this.filterHash.length; i++)
709 if (this.filterHash[i] == undefined) continue;
710 if (filter > 'pz:id=') filter = filter + '|';
711 filter += this.filterHash[i].id;
715 totalLength: function()
717 var simpleLength = this.simpleQuery != '' ? 1 : 0;
718 return this.advTerms.length + simpleLength;
720 clearSingleFilter: function()
722 this.singleFilter = null;
724 setSingleFilter: function(name, value)
726 this.singleFilter = {"name": name, "id": value };
728 getSingleFilterName: function()
730 return this.singleFilter.name;