From 25f8409e0007a1169c945be0591d2b4987b534b0 Mon Sep 17 00:00:00 2001 From: Mike Taylor Date: Fri, 14 Jun 2013 16:51:40 +0100 Subject: [PATCH] Initially copied from ../spdemo --- experiments/spclient/example_client.js | 394 +++++++++++ experiments/spclient/favicon.ico | Bin 0 -> 198 bytes experiments/spclient/index.html | 112 ++++ experiments/spclient/indexdata_logo.png | Bin 0 -> 3824 bytes experiments/spclient/pz2.js | 1110 +++++++++++++++++++++++++++++++ experiments/spclient/robots.txt | 2 + experiments/spclient/spclient.html | 30 - experiments/spclient/styles.css | 110 +++ 8 files changed, 1728 insertions(+), 30 deletions(-) create mode 100644 experiments/spclient/example_client.js create mode 100644 experiments/spclient/favicon.ico create mode 100644 experiments/spclient/index.html create mode 100644 experiments/spclient/indexdata_logo.png create mode 100644 experiments/spclient/pz2.js create mode 100644 experiments/spclient/robots.txt delete mode 100644 experiments/spclient/spclient.html create mode 100644 experiments/spclient/styles.css diff --git a/experiments/spclient/example_client.js b/experiments/spclient/example_client.js new file mode 100644 index 0000000..a8b9a9a --- /dev/null +++ b/experiments/spclient/example_client.js @@ -0,0 +1,394 @@ +/* A very simple client that shows a basic usage of the pz2.js +*/ + +// create a parameters array and pass it to the pz2's constructor +// then register the form submit event with the pz2.search function +// autoInit is set to true on default + +var pazpar2URL = "/pazpar2/search.pz2"; +var serviceProxyURL = "/service-proxy/"; +var authURLServiceProxy = "/service-proxy-auth"; +var pazpar2path = useServiceProxy ? serviceProxyURL : pazpar2URL; + +var usesessions; + +var showResponseType = ''; +if (document.location.hash == '#useproxy') { + usesessions = false; + pazpar2path = '/service-proxy/'; + showResponseType = 'json'; +} +if (useServiceProxy) { + usesessions = false; +} + +var my_paz = new pz2( { "onshow": my_onshow, + "showtime": 500, //each timer (show, stat, term, bytarget) can be specified this way + "pazpar2path": pazpar2path, + "oninit": my_oninit, + "onstat": my_onstat, + "onterm": my_onterm, + "termlist": "xtargets,subject,author", + "onbytarget": my_onbytarget, + "usesessions" : usesessions, + "showResponseType": showResponseType, + "onrecord": my_onrecord } ); +// some state vars +var curPage = 1; +var recPerPage = 20; +var totalRec = 0; +var curDetRecId = ''; +var curDetRecData = null; +var curSort = 'relevance'; +var curFilter = null; +var submitted = false; +var SourceMax = 16; +var SubjectMax = 10; +var AuthorMax = 10; + +// +// pz2.js event handlers: +// +function my_oninit() { + my_paz.stat(); + my_paz.bytarget(); +} + +function my_onshow(data) { + totalRec = data.merged; + // move it out + var pager = document.getElementById("pager"); + pager.innerHTML = ""; + pager.innerHTML +='
Displaying: ' + + (data.start + 1) + ' to ' + (data.start + data.num) + + ' of ' + data.merged + ' (found: ' + + data.total + ')
'; + drawPager(pager); + // navi + var results = document.getElementById("results"); + + var html = []; + for (var i = 0; i < data.hits.length; i++) { + var hit = data.hits[i]; + html.push('
' + +''+ (i + 1 + recPerPage * (curPage - 1)) +'. ' + +'' + + hit["md-title"] +' '); + if (hit["md-title-remainder"] !== undefined) { + html.push('' + hit["md-title-remainder"] + ' '); + } + if (hit["md-title-responsibility"] !== undefined) { + html.push(''+hit["md-title-responsibility"]+''); + } + if (hit.recid == curDetRecId) { + html.push(renderDetails(curDetRecData)); + } + html.push('
'); + } + replaceHtml(results, html.join('')); +} + +function my_onstat(data) { + var stat = document.getElementById("stat"); + if (stat == null) + return; + + stat.innerHTML = ' .:STATUS INFO -- Active clients: ' + + data.activeclients + + '/' + data.clients + ' -- ' + + 'Retrieved records: ' + data.records + + '/' + data.hits + + ' -- by ' + + (useServiceProxy ? 'service proxy' : 'pazpar2') + + ' :.'; +} + +function my_onterm(data) { + var termlists = []; + termlists.push('
TERMLISTS:
.::Sources
'); + for (var i = 0; i < data.xtargets.length && i < SourceMax; i++ ) { + termlists.push('' + data.xtargets[i].name + + ' (' + data.xtargets[i].freq + ')
'); + } + + termlists.push('
.::Subjects
'); + for (var i = 0; i < data.subject.length && i < SubjectMax; i++ ) { + termlists.push('' + data.subject[i].name + ' (' + + data.subject[i].freq + ')
'); + } + + termlists.push('
.::Authors
'); + for (var i = 0; i < data.author.length && i < AuthorMax; i++ ) { + termlists.push('' + + data.author[i].name + + ' (' + + data.author[i].freq + + ')
'); + } + var termlist = document.getElementById("termlist"); + replaceHtml(termlist, termlists.join('')); +} + +function my_onrecord(data) { + // FIXME: record is async!! + clearTimeout(my_paz.recordTimer); + // in case on_show was faster to redraw element + var detRecordDiv = document.getElementById('det_'+data.recid); + if (detRecordDiv) return; + curDetRecData = data; + var recordDiv = document.getElementById('recdiv_'+curDetRecData.recid); + var html = renderDetails(curDetRecData); + recordDiv.innerHTML += html; +} + +function my_onbytarget(data) { + var targetDiv = document.getElementById("bytarget"); + var table ='' + +''; + + for (var i = 0; i < data.length; i++ ) { + table += ""; + } + + table += '
Target IDHitsDiagsRecordsState
" + data[i].id + + "" + data[i].hits + + "" + data[i].diagnostic + + "" + data[i].records + + "" + data[i].state + "
'; + targetDiv.innerHTML = table; +} + +//////////////////////////////////////////////////////////////////////////////// +//////////////////////////////////////////////////////////////////////////////// + +// wait until the DOM is ready +function domReady () +{ + document.search.onsubmit = onFormSubmitEventHandler; + document.search.query.value = ''; + document.select.sort.onchange = onSelectDdChange; + document.select.perpage.onchange = onSelectDdChange; +} + +// when search button pressed +function onFormSubmitEventHandler() +{ + resetPage(); + loadSelect(); + triggerSearch(); + submitted = true; + return false; +} + +function onSelectDdChange() +{ + if (!submitted) return false; + resetPage(); + loadSelect(); + my_paz.show(0, recPerPage, curSort); + return false; +} + +function resetPage() +{ + curPage = 1; + totalRec = 0; +} + +function triggerSearch () +{ + my_paz.search(document.search.query.value, recPerPage, curSort, curFilter); +} + +function loadSelect () +{ + curSort = document.select.sort.value; + recPerPage = document.select.perpage.value; +} + +// limit the query after clicking the facet +function limitQuery (field, value) +{ + document.search.query.value += ' and ' + field + '="' + value + '"'; + onFormSubmitEventHandler(); +} + +// limit by target functions +function limitTarget (id, name) +{ + var navi = document.getElementById('navi'); + navi.innerHTML = + 'Source: ' + + name + ''; + navi.innerHTML += '
'; + curFilter = 'pz:id=' + id; + resetPage(); + loadSelect(); + triggerSearch(); + return false; +} + +function delimitTarget () +{ + var navi = document.getElementById('navi'); + navi.innerHTML = ''; + curFilter = null; + resetPage(); + loadSelect(); + triggerSearch(); + return false; +} + +function drawPager (pagerDiv) +{ + //client indexes pages from 1 but pz2 from 0 + var onsides = 6; + var pages = Math.ceil(totalRec / recPerPage); + + var firstClkbl = ( curPage - onsides > 0 ) + ? curPage - onsides + : 1; + + var lastClkbl = firstClkbl + 2*onsides < pages + ? firstClkbl + 2*onsides + : pages; + + var prev = '<< Prev | '; + if (curPage > 1) + prev = ' | '; + + var middle = ''; + for(var i = firstClkbl; i <= lastClkbl; i++) { + var numLabel = i; + if(i == curPage) + numLabel = '' + i + ''; + + middle += ' ' + + numLabel + ' '; + } + + var next = ' | Next >>'; + if (pages - curPage > 0) + next = ' | '; + + var predots = ''; + if (firstClkbl > 1) + predots = '...'; + + var postdots = ''; + if (lastClkbl < pages) + postdots = '...'; + + pagerDiv.innerHTML += '
' + + prev + predots + middle + postdots + next + '

'; +} + +function showPage (pageNum) +{ + curPage = pageNum; + my_paz.showPage( curPage - 1 ); +} + +// simple paging functions + +function pagerNext() { + if ( totalRec - recPerPage*curPage > 0) { + my_paz.showNext(); + curPage++; + } +} + +function pagerPrev() { + if ( my_paz.showPrev() != false ) + curPage--; +} + +// swithing view between targets and records + +function switchView(view) { + + var targets = document.getElementById('targetview'); + var records = document.getElementById('recordview'); + + switch(view) { + case 'targetview': + targets.style.display = "block"; + records.style.display = "none"; + break; + case 'recordview': + targets.style.display = "none"; + records.style.display = "block"; + break; + default: + alert('Unknown view.'); + } +} + +// detailed record drawing +function showDetails (prefixRecId) { + var recId = prefixRecId.replace('rec_', ''); + var oldRecId = curDetRecId; + curDetRecId = recId; + + // remove current detailed view if any + var detRecordDiv = document.getElementById('det_'+oldRecId); + // lovin DOM! + if (detRecordDiv) + detRecordDiv.parentNode.removeChild(detRecordDiv); + + // if the same clicked, just hide + if (recId == oldRecId) { + curDetRecId = ''; + curDetRecData = null; + return; + } + // request the record + my_paz.record(recId); +} + +function replaceHtml(el, html) { + var oldEl = typeof el === "string" ? document.getElementById(el) : el; + /*@cc_on // Pure innerHTML is slightly faster in IE + oldEl.innerHTML = html; + return oldEl; + @*/ + var newEl = oldEl.cloneNode(false); + newEl.innerHTML = html; + oldEl.parentNode.replaceChild(newEl, oldEl); + /* Since we just removed the old element from the DOM, return a reference + to the new element, which can be used to restore variable references. */ + return newEl; +}; + +function renderDetails(data, marker) +{ + var details = '
'; + if (marker) details += ''; + if (data["md-title"] != undefined) { + details += ''; + } + if (data["md-date"] != undefined) + details += ''; + if (data["md-author"] != undefined) + details += ''; + if (data["md-electronic-url"] != undefined) + details += ''; + if (data["location"][0]["md-subject"] != undefined) + details += ''; + if (data["location"][0]["@name"] != undefined) + details += ''; + details += '
'+ marker + '
Title: '+data["md-title"]; + if (data["md-title-remainder"] !== undefined) { + details += ' : ' + data["md-title-remainder"] + ' '; + } + if (data["md-title-responsibility"] !== undefined) { + details += ' '+ data["md-title-responsibility"] +''; + } + details += '
Date: ' + data["md-date"] + '
Author: ' + data["md-author"] + '
URL: ' + data["md-electronic-url"] + '' + '
Subject: ' + data["location"][0]["md-subject"] + '
Location: ' + data["location"][0]["@name"] + " (" +data["location"][0]["@id"] + ")" + '
'; + return details; +} + //EOF diff --git a/experiments/spclient/favicon.ico b/experiments/spclient/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..35c4899610de57317a1a22ba1610407ff738e370 GIT binary patch literal 198 zcmZQzU<5(|0VXiLfq{WR42U&=SOAC>fEXl?3#6SfX88aAKZ-Q-0R{$a7^Vm&2jf$N F1^|ad3&8*Y literal 0 HcmV?d00001 diff --git a/experiments/spclient/index.html b/experiments/spclient/index.html new file mode 100644 index 0000000..0954f03 --- /dev/null +++ b/experiments/spclient/index.html @@ -0,0 +1,112 @@ + + + + + + + Pazpar2 demo client + + + + + + + + + + + + + + + +
+ +
+ +
+ Record Browser + | + Target Info +
+ +
+ + + + + + +
Pazpar2 +
+ + +
+
+ +
+
+ +
+ + + + + +
+
+
+
+
+ Sort by + + and show + + per page. +
+
+
+ +
+
+
+ + + + + + + diff --git a/experiments/spclient/indexdata_logo.png b/experiments/spclient/indexdata_logo.png new file mode 100644 index 0000000000000000000000000000000000000000..171c03986a1b846dc1bf12d255f57e62aad2df68 GIT binary patch literal 3824 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 mT4625YB1_Z=Lto#oT%Cs^MI#bjD0000 0) { + context.keepAlive = data.getElementsByTagName("keepAlive")[0].childNodes[0].nodeValue; + } + context.pingTimer = + setTimeout( + function () { + context.ping(); + }, + context.keepAlive + ); + if ( context.initCallback ) + context.initCallback(); + } + else + context.throwError('Init failed. Malformed WS resonse.', + 110); + } + ); + // when through proxy no need to init + } else { + this.initStatusOK = true; + } + }, + // no need to ping explicitly + ping: function () + { + // pinging only makes sense when using pazpar2 directly + if( !this.initStatusOK || !this.useSessions ) + throw new Error( + 'Pz2.js: Ping not allowed (proxy mode) or session not initialized.' + ); + var context = this; + + clearTimeout(context.pingTimer); + + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + request.safeGet( + { "command": "ping", "session": this.sessionID, "windowid" : window.name }, + function(data) { + if ( data.getElementsByTagName("status")[0] + .childNodes[0].nodeValue == "OK" ) { + context.pingStatusOK = true; + context.pingTimer = + setTimeout( + function () { + context.ping(); + }, + context.keepAlive + ); + } + else + context.throwError('Ping failed. Malformed WS resonse.', + 111); + } + ); + }, + search: function (query, num, sort, filter, showfrom, addParamsArr) + { + clearTimeout(this.statTimer); + clearTimeout(this.showTimer); + clearTimeout(this.termTimer); + clearTimeout(this.bytargetTimer); + + this.showCounter = 0; + this.termCounter = 0; + this.bytargetCounter = 0; + this.statCounter = 0; + this.activeClients = 1; + + // no proxy mode + if( !this.initStatusOK ) + throw new Error('Pz2.js: session not initialized.'); + + if( query !== undefined ) + this.currQuery = query; + else + throw new Error("Pz2.js: no query supplied to the search command."); + + if ( showfrom !== undefined ) + var start = showfrom; + else + var start = 0; + + var searchParams = { + "command": "search", + "query": this.currQuery, + "session": this.sessionID, + "windowid" : window.name + }; + + if( sort !== undefined ) { + this.currentSort = sort; + searchParams["sort"] = sort; + } + if (filter !== undefined) + searchParams["filter"] = filter; + + // copy additional parmeters, do not overwrite + if (addParamsArr != undefined) { + for (var prop in addParamsArr) { + if (!searchParams.hasOwnProperty(prop)) + searchParams[prop] = addParamsArr[prop]; + } + } + + var context = this; + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + request.safeGet( + searchParams, + function(data) { + if ( data.getElementsByTagName("status")[0] + .childNodes[0].nodeValue == "OK" ) { + context.searchStatusOK = true; + //piggyback search + context.show(start, num, sort); + if (context.statCallback) + context.stat(); + if (context.termlistCallback) + context.termlist(); + if (context.bytargetCallback) + context.bytarget(); + } + else + context.throwError('Search failed. Malformed WS resonse.', + 112); + } + ); + }, + stat: function() + { + if( !this.initStatusOK ) + throw new Error('Pz2.js: session not initialized.'); + + // if called explicitly takes precedence + clearTimeout(this.statTimer); + + var context = this; + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + request.safeGet( + { "command": "stat", "session": this.sessionID, "windowid" : window.name }, + function(data) { + if ( data.getElementsByTagName("stat") ) { + var activeClients = + Number( data.getElementsByTagName("activeclients")[0] + .childNodes[0].nodeValue ); + context.activeClients = activeClients; + + var stat = Element_parseChildNodes(data.documentElement); + + context.statCounter++; + var delay = context.statTime + + context.statCounter * context.dumpFactor; + + if ( activeClients > 0 ) + context.statTimer = + setTimeout( + function () { + context.stat(); + }, + delay + ); + context.statCallback(stat); + } + else + context.throwError('Stat failed. Malformed WS resonse.', + 113); + } + ); + }, + show: function(start, num, sort, query_state) + { + if( !this.searchStatusOK && this.useSessions ) + throw new Error( + 'Pz2.js: show command has to be preceded with a search command.' + ); + + // if called explicitly takes precedence + clearTimeout(this.showTimer); + + if( sort !== undefined ) + this.currentSort = sort; + if( start !== undefined ) + this.currentStart = Number( start ); + if( num !== undefined ) + this.currentNum = Number( num ); + + var context = this; + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + var requestParameters = + { + "command": "show", + "session": this.sessionID, + "start": this.currentStart, + "num": this.currentNum, + "sort": this.currentSort, + "block": 1, + "type": this.showResponseType, + "windowid" : window.name + }; + if (query_state) + requestParameters["query-state"] = query_state; + if (this.version && this.version > 0) + requestParameters["version"] = this.version; + request.safeGet( + requestParameters, + function(data, type) { + var show = null; + var activeClients = 0; + if (type === "json") { + show = {}; + activeClients = Number(data.activeclients[0]); + show.activeclients = activeClients; + show.merged = Number(data.merged[0]); + show.total = Number(data.total[0]); + show.start = Number(data.start[0]); + show.num = Number(data.num[0]); + show.hits = data.hit; + } else if (data.getElementsByTagName("status")[0] + .childNodes[0].nodeValue == "OK") { + // first parse the status data send along with records + // this is strictly bound to the format + activeClients = + Number(data.getElementsByTagName("activeclients")[0] + .childNodes[0].nodeValue); + show = { + "activeclients": activeClients, + "merged": + Number( data.getElementsByTagName("merged")[0] + .childNodes[0].nodeValue ), + "total": + Number( data.getElementsByTagName("total")[0] + .childNodes[0].nodeValue ), + "start": + Number( data.getElementsByTagName("start")[0] + .childNodes[0].nodeValue ), + "num": + Number( data.getElementsByTagName("num")[0] + .childNodes[0].nodeValue ), + "hits": [] + }; + // parse all the first-level nodes for all tags + var hits = data.getElementsByTagName("hit"); + for (i = 0; i < hits.length; i++) + show.hits[i] = Element_parseChildNodes(hits[i]); + } else { + context.throwError('Show failed. Malformed WS resonse.', + 114); + }; + + var approxNode = data.getElementsByTagName("approximation"); + if (approxNode && approxNode[0] && approxNode[0].childNodes[0] && approxNode[0].childNodes[0].nodeValue) + show['approximation'] = + Number( approxNode[0].childNodes[0].nodeValue); + + + data.getElementsByTagName("") + context.activeClients = activeClients; + context.showCounter++; + var delay = context.showTime; + if (context.showCounter > context.showFastCount) + delay += context.showCounter * context.dumpFactor; + if ( activeClients > 0 ) + context.showTimer = setTimeout( + function () { + context.show(); + }, + delay); + context.showCallback(show); + } + ); + }, + record: function(id, offset, syntax, handler) + { + // we may call record with no previous search if in proxy mode + if(!this.searchStatusOK && this.useSessions) + throw new Error( + 'Pz2.js: record command has to be preceded with a search command.' + ); + + if( id !== undefined ) + this.currRecID = id; + + var recordParams = { + "command": "record", + "session": this.sessionID, + "id": this.currRecID, + "windowid" : window.name + }; + + this.currRecOffset = null; + if (offset != undefined) { + recordParams["offset"] = offset; + this.currRecOffset = offset; + } + + if (syntax != undefined) + recordParams['syntax'] = syntax; + + //overwrite default callback id needed + var callback = this.recordCallback; + var args = undefined; + if (handler != undefined) { + callback = handler['callback']; + args = handler['args']; + } + + var context = this; + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + + request.safeGet( + recordParams, + function(data) { + var recordNode; + var record; + //raw record + if (context.currRecOffset !== null) { + record = new Array(); + record['xmlDoc'] = data; + record['offset'] = context.currRecOffset; + callback(record, args); + //pz2 record + } else if ( recordNode = + data.getElementsByTagName("record")[0] ) { + // if stylesheet was fetched do not parse the response + if ( context.xslDoc ) { + record = new Array(); + record['xmlDoc'] = data; + record['xslDoc'] = context.xslDoc; + record['recid'] = + recordNode.getElementsByTagName("recid")[0] + .firstChild.nodeValue; + //parse record + } else { + record = Element_parseChildNodes(recordNode); + } + var activeClients = + Number( data.getElementsByTagName("activeclients")[0] + .childNodes[0].nodeValue ); + context.activeClients = activeClients; + context.recordCounter++; + var delay = context.recordTime + context.recordCounter * context.dumpFactor; + if ( activeClients > 0 ) + context.recordTimer = + setTimeout ( + function() { + context.record(id, offset, syntax, handler); + }, + delay + ); + callback(record, args); + } + else + context.throwError('Record failed. Malformed WS resonse.', + 115); + } + ); + }, + + termlist: function() + { + if( !this.searchStatusOK && this.useSessions ) + throw new Error( + 'Pz2.js: termlist command has to be preceded with a search command.' + ); + + // if called explicitly takes precedence + clearTimeout(this.termTimer); + + var context = this; + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + request.safeGet( + { + "command": "termlist", + "session": this.sessionID, + "name": this.termKeys, + "windowid" : window.name, + "version" : this.version + + }, + function(data) { + if ( data.getElementsByTagName("termlist") ) { + var activeClients = + Number( data.getElementsByTagName("activeclients")[0] + .childNodes[0].nodeValue ); + context.activeClients = activeClients; + var termList = { "activeclients": activeClients }; + var termLists = data.getElementsByTagName("list"); + //for each termlist + for (i = 0; i < termLists.length; i++) { + var listName = termLists[i].getAttribute('name'); + termList[listName] = new Array(); + var terms = termLists[i].getElementsByTagName('term'); + //for each term in the list + for (j = 0; j < terms.length; j++) { + var term = { + "name": + (terms[j].getElementsByTagName("name")[0] + .childNodes.length + ? terms[j].getElementsByTagName("name")[0] + .childNodes[0].nodeValue + : 'ERROR'), + "freq": + terms[j] + .getElementsByTagName("frequency")[0] + .childNodes[0].nodeValue || 'ERROR' + }; + + // Only for xtargets: id, records, filtered + var termIdNode = + terms[j].getElementsByTagName("id"); + if(terms[j].getElementsByTagName("id").length) + term["id"] = + termIdNode[0].childNodes[0].nodeValue; + termList[listName][j] = term; + + var recordsNode = terms[j].getElementsByTagName("records"); + if (recordsNode && recordsNode.length) + term["records"] = recordsNode[0].childNodes[0].nodeValue; + + var filteredNode = terms[j].getElementsByTagName("filtered"); + if (filteredNode && filteredNode.length) + term["filtered"] = filteredNode[0].childNodes[0].nodeValue; + + } + } + + context.termCounter++; + var delay = context.termTime + + context.termCounter * context.dumpFactor; + if ( activeClients > 0 ) + context.termTimer = + setTimeout( + function () { + context.termlist(); + }, + delay + ); + + context.termlistCallback(termList); + } + else + context.throwError('Termlist failed. Malformed WS resonse.', + 116); + } + ); + + }, + bytarget: function() + { + if( !this.initStatusOK && this.useSessions ) + throw new Error( + 'Pz2.js: bytarget command has to be preceded with a search command.' + ); + + // no need to continue + if( !this.searchStatusOK ) + return; + + // if called explicitly takes precedence + clearTimeout(this.bytargetTimer); + + var context = this; + var request = new pzHttpRequest(this.pz2String, this.errorHandler); + request.safeGet( + { + "command": "bytarget", + "session": this.sessionID, + "block": 1, + "windowid" : window.name, + "version" : this.version + }, + function(data) { + if ( data.getElementsByTagName("status")[0] + .childNodes[0].nodeValue == "OK" ) { + var targetNodes = data.getElementsByTagName("target"); + var bytarget = new Array(); + for ( i = 0; i < targetNodes.length; i++) { + bytarget[i] = new Array(); + for( j = 0; j < targetNodes[i].childNodes.length; j++ ) { + if ( targetNodes[i].childNodes[j].nodeType + == Node.ELEMENT_NODE ) { + var nodeName = + targetNodes[i].childNodes[j].nodeName; + if (targetNodes[i].childNodes[j].firstChild != null) + { + var nodeText = targetNodes[i].childNodes[j] + .firstChild.nodeValue; + bytarget[i][nodeName] = nodeText; + } + else { + bytarget[i][nodeName] = ""; + } + + + } + } + if (bytarget[i]["state"]=="Client_Disconnected") { + bytarget[i]["hits"] = "Error"; + } else if (bytarget[i]["state"]=="Client_Error") { + bytarget[i]["hits"] = "Error"; + } else if (bytarget[i]["state"]=="Client_Working") { + bytarget[i]["hits"] = "..."; + } + if (bytarget[i].diagnostic == "1") { + bytarget[i].diagnostic = "Permanent system error"; + } else if (bytarget[i].diagnostic == "2") { + bytarget[i].diagnostic = "Temporary system error"; + } + var targetsSuggestions = targetNodes[i].getElementsByTagName("suggestions"); + if (targetsSuggestions != undefined && targetsSuggestions.length>0) { + var suggestions = targetsSuggestions[0]; + bytarget[i]["suggestions"] = Element_parseChildNodes(suggestions); + } + } + + context.bytargetCounter++; + var delay = context.bytargetTime + + context.bytargetCounter * context.dumpFactor; + if ( context.activeClients > 0 ) + context.bytargetTimer = + setTimeout( + function () { + context.bytarget(); + }, + delay + ); + + context.bytargetCallback(bytarget); + } + else + context.throwError('Bytarget failed. Malformed WS resonse.', + 117); + } + ); + }, + + // just for testing, probably shouldn't be here + showNext: function(page) + { + var step = page || 1; + this.show( ( step * this.currentNum ) + this.currentStart ); + }, + + showPrev: function(page) + { + if (this.currentStart == 0 ) + return false; + var step = page || 1; + var newStart = this.currentStart - (step * this.currentNum ); + this.show( newStart > 0 ? newStart : 0 ); + }, + + showPage: function(pageNum) + { + //var page = pageNum || 1; + this.show(pageNum * this.currentNum); + } +}; + +/* +******************************************************************************** +** AJAX HELPER CLASS *********************************************************** +******************************************************************************** +*/ +var pzHttpRequest = function ( url, errorHandler ) { + this.maxUrlLength = 2048; + this.request = null; + this.url = url; + this.errorHandler = errorHandler || null; + this.async = true; + this.requestHeaders = {}; + + if ( window.XMLHttpRequest ) { + this.request = new XMLHttpRequest(); + } else if ( window.ActiveXObject ) { + try { + this.request = new ActiveXObject( 'Msxml2.XMLHTTP' ); + } catch (err) { + this.request = new ActiveXObject( 'Microsoft.XMLHTTP' ); + } + } +}; + + +pzHttpRequest.prototype = +{ + safeGet: function ( params, callback ) + { + var encodedParams = this.encodeParams(params); + var url = this._urlAppendParams(encodedParams); + if (url.length >= this.maxUrlLength) { + this.requestHeaders["Content-Type"] + = "application/x-www-form-urlencoded"; + this._send( 'POST', this.url, encodedParams, callback ); + } else { + this._send( 'GET', url, '', callback ); + } + }, + + get: function ( params, callback ) + { + this._send( 'GET', this._urlAppendParams(this.encodeParams(params)), + '', callback ); + }, + + post: function ( params, data, callback ) + { + this._send( 'POST', this._urlAppendParams(this.encodeParams(params)), + data, callback ); + }, + + load: function () + { + this.async = false; + this.request.open( 'GET', this.url, this.async ); + this.request.send(''); + if ( this.request.status == 200 ) + return this.request.responseXML; + }, + + encodeParams: function (params) + { + var sep = ""; + var encoded = ""; + for (var key in params) { + if (params[key] != null) { + encoded += sep + key + '=' + encodeURIComponent(params[key]); + sep = '&'; + } + } + return encoded; + }, + + _send: function ( type, url, data, callback) + { + var context = this; + this.callback = callback; + this.async = true; + this.request.open( type, url, this.async ); + for (var key in this.requestHeaders) + this.request.setRequestHeader(key, this.requestHeaders[key]); + this.request.onreadystatechange = function () { + context._handleResponse(url); /// url used ONLY for error reporting + } + this.request.send(data); + }, + + _urlAppendParams: function (encodedParams) + { + if (encodedParams) + return this.url + "?" + encodedParams; + else + return this.url; + }, + + _handleResponse: function (savedUrlForErrorReporting) + { + if ( this.request.readyState == 4 ) { + // pick up appplication errors first + var errNode = null; + if (this.request.responseXML && + (errNode = this.request.responseXML.documentElement) + && errNode.nodeName == 'error') { + var errMsg = errNode.getAttribute("msg"); + var errCode = errNode.getAttribute("code"); + var errAddInfo = ''; + if (errNode.childNodes.length) + errAddInfo = ': ' + errNode.childNodes[0].nodeValue; + + var err = new Error(errMsg + errAddInfo); + err.code = errCode; + + if (this.errorHandler) { + this.errorHandler(err); + } + else { + throw err; + } + } else if (this.request.status == 200 && + this.request.responseXML == null) { + if (this.request.responseText != null) { + //assume JSON + + var json = null; + var text = this.request.responseText; + if (typeof window.JSON == "undefined") + json = eval("(" + text + ")"); + else { + try { + json = JSON.parse(text); + } + catch (e) { + // Safari: eval will fail as well. Considering trying JSON2 (non-native implementation) instead + /* DEBUG only works in mk2-mobile + if (document.getElementById("log")) + document.getElementById("log").innerHTML = "" + e + " " + length + ": " + text; + */ + try { + json = eval("(" + text + ")"); + } + catch (e) { + /* DEBUG only works in mk2-mobile + if (document.getElementById("log")) + document.getElementById("log").innerHTML = "" + e + " " + length + ": " + text; + */ + } + } + } + this.callback(json, "json"); + } else { + var err = new Error("XML response is empty but no error " + + "for " + savedUrlForErrorReporting); + err.code = -1; + if (this.errorHandler) { + this.errorHandler(err); + } else { + throw err; + } + } + } else if (this.request.status == 200) { + this.callback(this.request.responseXML); + } else { + var err = new Error("HTTP response not OK: " + + this.request.status + " - " + + this.request.statusText ); + err.code = '00' + this.request.status; + if (this.errorHandler) { + this.errorHandler(err); + } + else { + throw err; + } + } + } + } +}; + +/* +******************************************************************************** +** XML HELPER FUNCTIONS ******************************************************** +******************************************************************************** +*/ + +// DOMDocument + +if ( window.ActiveXObject) { + var DOMDoc = document; +} else { + var DOMDoc = Document.prototype; +} + +DOMDoc.newXmlDoc = function ( root ) +{ + var doc; + + if (document.implementation && document.implementation.createDocument) { + doc = document.implementation.createDocument('', root, null); + } else if ( window.ActiveXObject ) { + doc = new ActiveXObject("MSXML2.DOMDocument"); + doc.loadXML('<' + root + '/>'); + } else { + throw new Error ('No XML support in this browser'); + } + + return doc; +} + + +DOMDoc.parseXmlFromString = function ( xmlString ) +{ + var doc; + + if ( window.DOMParser ) { + var parser = new DOMParser(); + doc = parser.parseFromString( xmlString, "text/xml"); + } else if ( window.ActiveXObject ) { + doc = new ActiveXObject("MSXML2.DOMDocument"); + doc.loadXML( xmlString ); + } else { + throw new Error ("No XML parsing support in this browser."); + } + + return doc; +} + +DOMDoc.transformToDoc = function (xmlDoc, xslDoc) +{ + if ( window.XSLTProcessor ) { + var proc = new XSLTProcessor(); + proc.importStylesheet( xslDoc ); + return proc.transformToDocument(xmlDoc); + } else if ( window.ActiveXObject ) { + return document.parseXmlFromString(xmlDoc.transformNode(xslDoc)); + } else { + alert( 'Unable to perform XSLT transformation in this browser' ); + } +} + +// DOMElement + +Element_removeFromDoc = function (DOM_Element) +{ + DOM_Element.parentNode.removeChild(DOM_Element); +} + +Element_emptyChildren = function (DOM_Element) +{ + while( DOM_Element.firstChild ) { + DOM_Element.removeChild( DOM_Element.firstChild ) + } +} + +Element_appendTransformResult = function ( DOM_Element, xmlDoc, xslDoc ) +{ + if ( window.XSLTProcessor ) { + var proc = new XSLTProcessor(); + proc.importStylesheet( xslDoc ); + var docFrag = false; + docFrag = proc.transformToFragment( xmlDoc, DOM_Element.ownerDocument ); + DOM_Element.appendChild(docFrag); + } else if ( window.ActiveXObject ) { + DOM_Element.innerHTML = xmlDoc.transformNode( xslDoc ); + } else { + alert( 'Unable to perform XSLT transformation in this browser' ); + } +} + +Element_appendTextNode = function (DOM_Element, tagName, textContent ) +{ + var node = DOM_Element.ownerDocument.createElement(tagName); + var text = DOM_Element.ownerDocument.createTextNode(textContent); + + DOM_Element.appendChild(node); + node.appendChild(text); + + return node; +} + +Element_setTextContent = function ( DOM_Element, textContent ) +{ + if (typeof DOM_Element.textContent !== "undefined") { + DOM_Element.textContent = textContent; + } else if (typeof DOM_Element.innerText !== "undefined" ) { + DOM_Element.innerText = textContent; + } else { + throw new Error("Cannot set text content of the node, no such method."); + } +} + +Element_getTextContent = function (DOM_Element) +{ + if ( typeof DOM_Element.textContent != 'undefined' ) { + return DOM_Element.textContent; + } else if (typeof DOM_Element.text != 'undefined') { + return DOM_Element.text; + } else { + throw new Error("Cannot get text content of the node, no such method."); + } +} + +Element_parseChildNodes = function (node) +{ + var parsed = {}; + var hasChildElems = false; + var textContent = ''; + + if (node.hasChildNodes()) { + var children = node.childNodes; + for (var i = 0; i < children.length; i++) { + var child = children[i]; + switch (child.nodeType) { + case Node.ELEMENT_NODE: + hasChildElems = true; + var nodeName = child.nodeName; + if (!(nodeName in parsed)) + parsed[nodeName] = []; + parsed[nodeName].push(Element_parseChildNodes(child)); + break; + case Node.TEXT_NODE: + textContent += child.nodeValue; + break; + case Node.CDATA_SECTION_NODE: + textContent += child.nodeValue; + break; + } + } + } + + var attrs = node.attributes; + for (var i = 0; i < attrs.length; i++) { + hasChildElems = true; + var attrName = '@' + attrs[i].nodeName; + var attrValue = attrs[i].nodeValue; + parsed[attrName] = attrValue; + } + + // if no nested elements/attrs set value to text + if (hasChildElems) + parsed['#text'] = textContent; + else + parsed = textContent; + + return parsed; +} + +/* do not remove trailing bracket */ +} diff --git a/experiments/spclient/robots.txt b/experiments/spclient/robots.txt new file mode 100644 index 0000000..1f53798 --- /dev/null +++ b/experiments/spclient/robots.txt @@ -0,0 +1,2 @@ +User-agent: * +Disallow: / diff --git a/experiments/spclient/spclient.html b/experiments/spclient/spclient.html deleted file mode 100644 index 08bb940..0000000 --- a/experiments/spclient/spclient.html +++ /dev/null @@ -1,30 +0,0 @@ - - - - - MasterKey Widget Set: SP Client - - - -

- JQuery UI demo. -

-

- Hello, world! -

-

- That seemed to go OK. -

- - diff --git a/experiments/spclient/styles.css b/experiments/spclient/styles.css new file mode 100644 index 0000000..e6cd2b0 --- /dev/null +++ b/experiments/spclient/styles.css @@ -0,0 +1,110 @@ +html { + overflow-y: scroll; + font-size: 12px; +} + +body { + font-family: tahoma, arial, sans-serif; + color: #156a16; +} + +hr { + border: 0; + color: #156a16; + background-color: #156a16; + height: 1px; +} + +thead { + font-weight: bold; +} + +a { + color: #005701; + text-decoration: none; +} + +a.extern { + color: #CC6600; +} + +a:hover { + text-decoration: underline; +} + +a.crossout:hover { + text-decoration: line-through; + font-weight: bold; +} + + +#heading { + background-color: #87c9af; + border-top: 1px solid #156a16; + color: #ffffff; + font-size: large; +} + +input#button { + border: 3px outset #132194; + background-color: #132194; + padding: 2px; + width: 6em; + color: #FFFFFF; + font-weight: bold; + text-transform: uppercase; + font-size: 10px; + margin-left: 8px; + cursor: pointer; +} + +input#query { + border: 2px inset #34cc67; + padding: 3px; + font-size: 12px; +} + +div.termtitle { + margin: 4px; + font-weight: bold; +} + +div.record { + padding: 5px; +} + +div.details { + border: 3px dashed gray; + color: gray; + padding: 5px; + margin: 4px; +} + +#switchmenu { + padding-bottom: 3px; + text-align: right; +} + +#recordview { + background-color: #fafafa; + border-bottom: 1px solid #156a16; +} + +#targetview { + background-color: #fafafa; + border-bottom: 1px solid #156a16; +} + +#bytarget { + padding: 7px; +} + +#footer { + padding-top: 4px; + color: #74c775; + text-align: center; +} + +#stat { + font-weight: bold; +} -- 1.7.10.4