498 lines
15 KiB
498 lines
15 KiB
14 years ago
|
var iiv = {};
|
||
|
|
||
|
iiv.Class = function(prototype) {
|
||
|
var c = function(options) {
|
||
|
jQuery.extend(this, options);
|
||
|
this.initialize.apply(this);
|
||
|
};
|
||
|
|
||
|
c.prototype = prototype;
|
||
|
return c;
|
||
|
};
|
||
|
|
||
|
iiv.Viewer = new iiv.Class({
|
||
|
ui: null,
|
||
|
iivContainer: '.iiv',
|
||
|
mapContainer: 'iiv-image-panel',
|
||
|
fedoraUrl: 'http://' + location.host + '/fedora',
|
||
|
pids: null,
|
||
|
pageIndex: 0,
|
||
|
map: null,
|
||
|
|
||
|
initialize: function(options) {
|
||
|
this.ui = new iiv.Viewer.UI({viewer: this});
|
||
|
this.riSearch = new iiv.Viewer.RISearch(this.riSearchOptions());
|
||
|
this.riSearch.search();
|
||
|
jQuery.iiv = this;
|
||
|
},
|
||
|
|
||
|
intercept: function(object, method, interceptor) {
|
||
|
object[method + '_without_interceptor'] = object[method];
|
||
|
object[method] = interceptor;
|
||
|
},
|
||
|
|
||
|
riSearchOptions: function() {
|
||
|
var viewer = this;
|
||
|
return {
|
||
|
pid: this.pid,
|
||
|
cmodel: this.cmodel,
|
||
|
dsid: this.dsid,
|
||
|
fedoraUrl: this.fedoraUrl,
|
||
|
uid: this.uid,
|
||
|
searchCallback: this.createSearchCallback()
|
||
|
}
|
||
|
},
|
||
|
|
||
|
createSearchCallback: function() {
|
||
|
var viewer = this;
|
||
|
|
||
|
return function(results) {
|
||
|
viewer.pids = results;
|
||
|
for (i = 0; i < viewer.pids.length; i++) {
|
||
|
if (viewer.pids[i] == viewer.pid) {
|
||
|
viewer.pageIndex = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
viewer.loadText();
|
||
|
viewer.initializeMap();
|
||
|
viewer.ui.initializeUI();
|
||
|
};
|
||
|
},
|
||
|
|
||
|
initializeMap: function() {
|
||
|
OpenLayers.Layer.OpenURL.viewerWidth = jQuery('.iiv-canvas').width();
|
||
|
OpenLayers.Layer.OpenURL.viewerHeight = jQuery('.iiv-canvas').height();
|
||
|
var imageLayer = this.createImageLayer();
|
||
|
var mapOptions = this.createMapOptions(imageLayer);
|
||
|
mapOptions.controls = this.createMapControls();
|
||
|
this.map = new OpenLayers.Map(this.mapContainer, mapOptions);
|
||
|
this.map.addLayer(imageLayer);
|
||
|
var lon = this.map.maxExtent.width / 2;
|
||
|
var lat = this.map.maxExtent.height / 2;
|
||
|
this.map.setCenter(new OpenLayers.LonLat(lon, lat), 0);
|
||
|
},
|
||
|
|
||
|
createMapControls: function() {
|
||
|
var controls = [
|
||
|
new OpenLayers.Control.MouseDefaults(),
|
||
|
new OpenLayers.Control.KeyboardDefaults()
|
||
|
];
|
||
|
|
||
|
return controls;
|
||
|
},
|
||
|
|
||
|
createMapOptions: function(imageLayer) {
|
||
|
var metadata = imageLayer.getImageMetadata();
|
||
|
var resolutions = imageLayer.getResolutions();
|
||
|
var maxExtent = new OpenLayers.Bounds(0, 0, metadata.width, metadata.height);
|
||
|
var tileSize = imageLayer.getTileSize();
|
||
|
return options = {resolutions: resolutions, maxExtent: maxExtent, tileSize: tileSize};
|
||
|
},
|
||
|
|
||
|
createImageLayer: function() {
|
||
|
var pid = this.currentPid();
|
||
|
var djatokaUrl = this.djatokaUrl(pid);
|
||
|
|
||
|
var imageLayer = new iiv.Viewer.ImageLayer('OpenURL', '', {
|
||
|
isBaseLayer : true,
|
||
|
layername : 'basic',
|
||
|
format : 'image/jpeg',
|
||
|
rft_id : this.rftUrl(pid),
|
||
|
metadataUrl : djatokaUrl + '/getMetadata?uid=' + this.uid
|
||
|
});
|
||
|
|
||
|
imageLayer.djatokaUrl = djatokaUrl;
|
||
|
imageLayer.uid = this.uid;
|
||
|
|
||
|
return imageLayer;
|
||
|
},
|
||
|
|
||
|
rftUrl: function(pid) {
|
||
|
return this.djatokaUrl(pid) + '/JP2?uid=djatoka';
|
||
|
},
|
||
|
|
||
|
currentPid: function() {
|
||
|
return this.pids[this.pageIndex];
|
||
|
},
|
||
|
|
||
|
djatokaUrl: function(pid) {
|
||
|
return this.pidUrl(pid) + '/ilives:jp2Sdef';
|
||
|
},
|
||
|
|
||
|
pidUrl: function(pid) {
|
||
|
return this.fedoraUrl + '/get/' + pid;
|
||
|
},
|
||
|
|
||
|
teiUrl: function(pid) {
|
||
|
return this.fedoraUrl + '/get/' + pid + '/ilives:tei2htmlSdef/tei2html?uid=' + this.uid;
|
||
|
},
|
||
|
|
||
|
setPage: function(index) {
|
||
|
if (index != this.pageIndex && index >= 0 && index < this.pids.length) {
|
||
|
this.pageIndex = index;
|
||
|
this.loadText();
|
||
|
|
||
|
var nextLayer = this.createImageLayer();
|
||
|
var options = this.createMapOptions(nextLayer);
|
||
|
this.map.resolutions = options.resolutions;
|
||
|
this.map.maxExtent = options.maxExtent;
|
||
|
this.map.tileSize = options.tileSize;
|
||
|
|
||
|
var baseLayer = this.map.baseLayer;
|
||
|
|
||
|
this.map.addLayer(nextLayer);
|
||
|
this.map.setBaseLayer(nextLayer);
|
||
|
this.map.removeLayer(baseLayer);
|
||
|
this.ui.updatePageControls(index);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
nextPid: function() {
|
||
|
this.setPage(this.pageIndex + 1);
|
||
|
},
|
||
|
|
||
|
previousPid: function() {
|
||
|
this.setPage(this.pageIndex - 1);
|
||
|
},
|
||
|
|
||
|
loadText: function() {
|
||
|
/* var container = this.ui.textContainer;
|
||
|
container.html('');
|
||
|
jQuery.get(this.teiUrl(this.currentPid()), function(data) {
|
||
|
container.html(data);
|
||
|
}, 'html');*/
|
||
|
},
|
||
|
|
||
|
getPrintUrl: function() {
|
||
|
var imageExtent = this.map.getMaxExtent();
|
||
|
var aspect = imageExtent.getWidth() / imageExtent.getHeight();
|
||
|
var scale = aspect > 1.3333 ? "600,0" : "0,800";
|
||
|
var level = '3'; // TODO calculate
|
||
|
|
||
|
// assemble url
|
||
|
var imageUrl = this.djatokaUrl(this.currentPid()) + '/getRegion?uid=' + this.uid + '&level=' + level + '&scale=' + scale;
|
||
|
var printUrl = '/iiv/print.html?pid=' + this.currentPid() + '&image=' + escape(imageUrl);
|
||
|
|
||
|
return printUrl;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
iiv.Viewer.UI = new iiv.Class({
|
||
|
viewer: null,
|
||
|
//sliderPage: null,
|
||
|
//buttonPagePrevious: null,
|
||
|
//buttonPageNext: null,
|
||
|
sliderZoom: null,
|
||
|
buttonZoomIn: null,
|
||
|
buttonZoomOut: null,
|
||
|
buttonZoomMax: null,
|
||
|
buttonText: null,
|
||
|
imagePanel: null,
|
||
|
//textPanel: null,
|
||
|
//textContainer: null,
|
||
|
buttonPrint: null,
|
||
|
|
||
|
initialize: function(options) {
|
||
|
this.createUI();
|
||
|
},
|
||
|
|
||
|
createDiv: function(parent, cssClass) {
|
||
|
var div = jQuery('<div class="' + cssClass + '"></div>');
|
||
|
parent.append(div);
|
||
|
return div;
|
||
|
},
|
||
|
|
||
|
createUI: function() {
|
||
|
var container = jQuery(this.viewer.iivContainer);
|
||
|
container.append('<link rel="stylesheet" href="/iiv/css/jquery-ui/smoothness/jquery-ui-1.7.2.custom.css" type="text/css" />');
|
||
|
container.append('<link rel="stylesheet" href="/iiv/css/iiv.css" type="text/css"/>');
|
||
|
container.append('<!--[if lte IE 6]><link rel="stylesheet" href="/iiv/css/ie6.css" type="text/css"><![endif]-->');
|
||
|
|
||
|
var ui = this.createDiv(container, 'iiv-ui ui-corner-all');
|
||
|
var toolbar = this.createDiv(ui, 'iiv-toolbar');
|
||
|
|
||
|
this.createZoomControls(toolbar);
|
||
|
//this.createPageControls(toolbar);
|
||
|
//this.createOtherControls(toolbar);
|
||
|
|
||
|
var canvas = this.createDiv(ui, 'iiv-canvas')
|
||
|
//this.textPanel = this.createDiv(canvas, 'iiv-text-panel');
|
||
|
//this.textContainer = this.createDiv(this.textPanel, 'iiv-text-container');
|
||
|
|
||
|
this.imagePanel = this.createDiv(canvas, 'iiv-image-panel');
|
||
|
this.imagePanel.attr('id', 'iiv-image-panel');
|
||
|
|
||
|
var clear = this.createDiv(container, 'iiv-clear');
|
||
|
|
||
|
jQuery('.ui-state-default').hover (
|
||
|
function(){
|
||
|
jQuery(this).addClass("ui-state-hover");
|
||
|
},
|
||
|
function(){
|
||
|
jQuery(this).removeClass("ui-state-hover");
|
||
|
}
|
||
|
);
|
||
|
},
|
||
|
|
||
|
createZoomControls: function(toolbar) {
|
||
|
var controls = this.createControlSet(toolbar, 'zoom');
|
||
|
this.buttonZoomIn = this.createButton(controls, 'zoom-in', 'Zoom in', 'ui-icon-zoomin');
|
||
|
this.buttonZoomOut = this.createButton(controls, 'zoom-out', 'Zoom out', 'ui-icon-zoomout');
|
||
|
this.buttonZoomMax = this.createButton(controls, 'zoom-max', 'Zoom to full image', 'ui-icon-search');
|
||
|
return controls;
|
||
|
},
|
||
|
|
||
|
createPageControls: function(toolbar) {
|
||
|
var controls = this.createControlSet(toolbar, 'page');
|
||
|
this.buttonPagePrevious = this.createButton(controls, 'page-previous', 'Previous page', 'ui-icon-arrowthick-1-w');
|
||
|
this.createPageNumberDisplay(controls);
|
||
|
this.buttonPageNext = this.createButton(controls, 'page-next', 'Next page', 'ui-icon-arrowthick-1-e');
|
||
|
return controls;
|
||
|
},
|
||
|
|
||
|
createOtherControls: function(toolbar) {
|
||
|
var controls = this.createControlSet(toolbar, 'other');
|
||
|
//this.buttonText = this.createButton(controls, 'text', 'Show text', 'iiv-icon-text');
|
||
|
this.buttonPrint = this.createButton(controls, 'print', 'Print page', 'ui-icon-print');
|
||
|
return controls;
|
||
|
},
|
||
|
|
||
|
createPageNumberDisplay: function(parent) {
|
||
|
var container = this.createDiv(parent, 'iiv-page-number');
|
||
|
this.currentPageSpan = jQuery('<span class="current">-</span>');
|
||
|
this.maxPageSpan = jQuery('<span class="max">-</span>');
|
||
|
container.append(this.currentPageSpan);
|
||
|
container.append('<span class="separator"> / </span>');
|
||
|
container.append(this.maxPageSpan);
|
||
|
return container;
|
||
|
},
|
||
|
|
||
|
createControlSet: function(parent, name) {
|
||
|
var controls = this.createDiv(parent, 'iiv-controlset ' + name);
|
||
|
parent.append(controls);
|
||
|
return controls;
|
||
|
},
|
||
|
|
||
|
createButton: function(parent, name, title, iconClass) {
|
||
|
var button = jQuery('<button class="' + name + ' ui-corner-all ui-state-default" title="' + title + '"><span class="ui-icon ' + iconClass + '"></span></button>');
|
||
|
parent.append(button);
|
||
|
return button;
|
||
|
},
|
||
|
|
||
|
|
||
|
initializeUI: function() {
|
||
|
this.addInterceptors();
|
||
|
|
||
|
this.updateZoomControls(this.viewer.map.getZoom());
|
||
|
|
||
|
this.addEventHandlers();
|
||
|
},
|
||
|
|
||
|
addInterceptors: function() {
|
||
|
var ui = this;
|
||
|
ui.viewer.intercept(this.viewer.map, 'setCenter', function(lonlat, zoom, dragging, forceZoomChange) {
|
||
|
if (zoom != null && zoom != ui.viewer.map.getZoom()) {
|
||
|
ui.updateZoomControls(zoom);
|
||
|
}
|
||
|
|
||
|
ui.viewer.map.setCenter_without_interceptor(lonlat, zoom, dragging, forceZoomChange);
|
||
|
});
|
||
|
},
|
||
|
|
||
|
addEventHandlers: function() {
|
||
|
var viewerUI = this;
|
||
|
viewerUI.buttonZoomIn.click(function() {
|
||
|
viewerUI.viewer.map.zoomIn();
|
||
|
});
|
||
|
|
||
|
viewerUI.buttonZoomOut.click(function() {
|
||
|
viewerUI.viewer.map.zoomOut();
|
||
|
});
|
||
|
|
||
|
viewerUI.buttonZoomMax.click(function() {
|
||
|
viewerUI.viewer.map.zoomToMaxExtent();
|
||
|
});
|
||
|
|
||
|
viewerUI.sliderZoom.bind('slidestop', function(event, ui) {
|
||
|
viewerUI.viewer.map.zoomTo(ui.value);
|
||
|
});
|
||
|
|
||
|
|
||
|
viewerUI.buttonPrint.click(function() {
|
||
|
viewerUI.printPage();
|
||
|
});
|
||
|
},
|
||
|
|
||
|
printPage: function() {
|
||
|
var url = this.viewer.getPrintUrl();
|
||
|
|
||
|
// open popup window
|
||
|
var popupWidth = Math.max(312, Math.min(624, window.screen.availWidth));
|
||
|
var popupHeight = Math.max(312, Math.min(824 / 2, window.screen.availHeight));
|
||
|
var features = 'width=' + popupWidth + ',height=' + popupHeight;
|
||
|
window.open(url, '_blank', features);
|
||
|
},
|
||
|
|
||
|
updatePageControls: function(page) {
|
||
|
this.sliderPage.slider('value', page)
|
||
|
this.currentPageSpan.text(page + 1);
|
||
|
|
||
|
if (page == 0) {
|
||
|
this.disable(this.buttonPagePrevious);
|
||
|
}
|
||
|
|
||
|
else {
|
||
|
this.enable(this.buttonPagePrevious);
|
||
|
}
|
||
|
|
||
|
if (page == this.sliderPage.slider('option', 'max')) {
|
||
|
this.disable(this.buttonPageNext);
|
||
|
}
|
||
|
|
||
|
else {
|
||
|
this.enable(this.buttonPageNext);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
updateZoomControls: function(zoom) {
|
||
|
|
||
|
this.enable(this.buttonZoomOut);
|
||
|
|
||
|
this.enable(this.buttonZoomIn);
|
||
|
},
|
||
|
|
||
|
disable: function(button) {
|
||
|
button.attr('disabled', 'disabled');
|
||
|
},
|
||
|
|
||
|
enable: function(button) {
|
||
|
button.removeAttr('disabled');
|
||
|
},
|
||
|
|
||
|
toggleText: function() {
|
||
|
this.buttonText.toggleClass('ui-state-active');
|
||
|
this.buttonText.toggleClass('ui-state-default');
|
||
|
this.imagePanel.toggleClass('narrow');
|
||
|
this.textPanel.toggle();
|
||
|
this.viewer.map.updateSize();
|
||
|
}
|
||
|
});
|
||
|
|
||
|
iiv.Viewer.RISearch = new iiv.Class({
|
||
|
type: 'tuples',
|
||
|
lang: 'itql',
|
||
|
format: 'csv',
|
||
|
query: null,
|
||
|
results: null,
|
||
|
|
||
|
initialize: function(options) {
|
||
|
if (!this.query) {
|
||
|
if (this.cmodel == 'ilives:bookCModel') {
|
||
|
this.query = 'select $object from <#ri> '
|
||
|
+ 'where ($object <fedora-model:hasModel> <fedora:ilives:pageCModel> '
|
||
|
+ 'and $object <fedora-rels-ext:isMemberOf> <fedora:' + this.pid + '>) '
|
||
|
+ 'order by $object';
|
||
|
}
|
||
|
|
||
|
else if (this.cmodel == 'ilives:pageCModel') {
|
||
|
this.query = 'select $parent '
|
||
|
+ 'subquery ('
|
||
|
+ ' select $sibling from <#ri> '
|
||
|
+ ' where $sibling <fedora-rels-ext:isMemberOf> $parent '
|
||
|
+ ' and $sibling <fedora-model:hasModel> <fedora:ilives:pageCModel> '
|
||
|
+ ' order by $sibling) '
|
||
|
+ 'from <#ri> '
|
||
|
+ 'where $child <mulgara:is> <fedora:' + this.pid + '> '
|
||
|
+ 'and $child <fedora-rels-ext:isMemberOf> $parent '
|
||
|
+ 'and $parent <fedora-model:hasModel> <fedora:ilives:bookCModel>';
|
||
|
} /*
|
||
|
else if (this.cmodel == 'islandora:slideCModel') {
|
||
|
this.query = 'select $parent '
|
||
|
+ 'subquery ('
|
||
|
+ ' select $sibling from <#ri> '
|
||
|
+ ' where $sibling <fedora-rels-ext:isMemberOfCollection> $parent '
|
||
|
+ ' and $sibling <fedora-model:hasModel> <fedora:islandora:slideCModel> '
|
||
|
+ ' order by $sibling) '
|
||
|
+ 'from <#ri> '
|
||
|
+ 'where $child <mulgara:is> <fedora:' + this.pid + '> '
|
||
|
+ 'and $child <fedora-rels-ext:isMemberOfCollection> $parent '
|
||
|
+ 'and $parent <fedora-model:hasModel> <fedora:islandora:collectionCModel>';
|
||
|
} */
|
||
|
else {
|
||
|
// no query -- pid will be used alone.
|
||
|
this.query = 'select $object from <#ri>'
|
||
|
+ 'where $object <mulgara:is> \'' + this.pid + '\'';
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
search: function() {
|
||
|
if (this.query == null) {
|
||
|
|
||
|
this.results = [this.pid];
|
||
|
}
|
||
|
|
||
|
else {
|
||
|
options = {
|
||
|
type: this.type,
|
||
|
lang: this.lang,
|
||
|
format: this.format,
|
||
|
query: this.query,
|
||
|
uid: this.uid
|
||
|
};
|
||
|
|
||
|
jQuery.post(this.fedoraUrl + '/risearch', options, this.createCallback(), 'text');
|
||
|
}
|
||
|
},
|
||
|
|
||
|
extractPid: function(riSearchResult) {
|
||
|
return riSearchResult.replace(/^.*\//, '');
|
||
|
},
|
||
|
|
||
|
createCallback: function() {
|
||
|
var riSearch = this;
|
||
|
|
||
|
return function(data, status) {
|
||
|
var results = [];
|
||
|
if ('success' == status) {
|
||
|
var lines = data.split("\n");
|
||
|
for (i = 0; i < lines.length; i++) {
|
||
|
if (i > 0 && lines[i] != '') {
|
||
|
results.push(riSearch.extractPid(lines[i]));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
riSearch.searchCallback(results);
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
|
||
|
/* monkey patch OpenLayers.Layer.OpenURL */
|
||
|
iiv.Viewer.ImageLayer = OpenLayers.Class(OpenLayers.Layer.OpenURL, {
|
||
|
djatokaUrl: null,
|
||
|
uid: null,
|
||
|
|
||
|
/**
|
||
|
* this implementation is the same as the superclass, except that we use a
|
||
|
* fedora service as the url base, not djatoka itself
|
||
|
*/
|
||
|
getURL: function(bounds) {
|
||
|
bounds = this.adjustBounds(bounds);
|
||
|
this.calculatePositionAndSize(bounds);
|
||
|
var z = this.map.getZoom() + this.zoomOffset;
|
||
|
|
||
|
// uid and djatokaUrl set in createImageLayer
|
||
|
var path = this.djatokaUrl + '/getRegion?uid=' + this.uid + '&level=' + z
|
||
|
+ '®ion=' + this.tilePos.lat + "," + this.tilePos.lon + "," + this.imageSize.h + "," + this.imageSize.w;
|
||
|
|
||
|
var url = this.url;
|
||
|
if (url instanceof Array) {
|
||
|
url = this.selectUrl(path, url);
|
||
|
}
|
||
|
return url + path;
|
||
|
}
|
||
|
});
|