// Error reporting - this helps us fix errors quickly function logError(description,page,line) { if (typeof(archive_analytics) != 'undefined') { var values = { 'bookreader': 'error', 'description': description, 'page': page, 'line': line, 'itemid': 'EinAstronomischerBeobachtungstextAusDem37.JahreNebukadnezarsIi', 'subPrefix': 'Nw1915OnVat4956-EnglishTranslationPdf', 'server': 'ia600800.us.archive.org', 'bookPath': '/22/items/EinAstronomischerBeobachtungstextAusDem37.JahreNebukadnezarsIi/Nw1915OnVat4956-EnglishTranslationPdf' }; // if no referrer set '-' as referrer if (document.referrer == '') { values['referrer'] = '-'; } else { values['referrer'] = document.referrer; } if (typeof(br) != 'undefined') { values['itemid'] = br.bookId; values['subPrefix'] = br.subPrefix; values['server'] = br.server; values['bookPath'] = br.bookPath; } var qs = archive_analytics.format_bug(values); var error_img = new Image(100,25); error_img.src = archive_analytics.img_src + "?" + qs; } return false; // allow browser error handling so user sees there was a problem } window.onerror=logError; br = new BookReader(); br.titleLeaf = 0; br.getPageWidth = function(index) { return this.pageW[index]; } br.getPageHeight = function(index) { return this.pageH[index]; } // Returns true if page image is available rotated br.canRotatePage = function(index) { return 'jp2' == this.imageFormat; // Assume single format for now } // reduce defaults to 1 (no reduction) // rotate defaults to 0 (no rotation) br.getPageURI = function(index, reduce, rotate) { var _reduce; var _rotate; if ('undefined' == typeof(reduce)) { _reduce = 1; } else { _reduce = reduce; } if ('undefined' == typeof(rotate)) { _rotate = 0; } else { _rotate = rotate; } var file = this._getPageFile(index); // $$$ add more image stack formats here return '//'+this.server+'/BookReader/BookReaderImages.php?zip='+this.zip+'&file='+file+'&scale='+_reduce+'&rotate='+_rotate; } // Get a rectangular region out of a page br.getRegionURI = function(index, reduce, rotate, sourceX, sourceY, sourceWidth, sourceHeight) { // Map function arguments to the url keys var urlKeys = ['n', 'r', 'rot', 'x', 'y', 'w', 'h']; var page = ''; for (var i = 0; i < arguments.length; i++) { if ('undefined' != typeof(arguments[i])) { if (i > 0 ) { page += '_'; } page += urlKeys[i] + arguments[i]; } } var itemPath = this.bookPath.replace(new RegExp('/'+this.subPrefix+'$'), ''); // remove trailing subPrefix return '//'+this.server+'/BookReader/BookReaderImages.php?id=' + this.bookId + '&itemPath=' + itemPath + '&server=' + this.server + '&subPrefix=' + this.subPrefix + '&page=' +page + '.jpg'; } br._getPageFile = function(index) { var leafStr = '0000'; var imgStr = this.leafMap[index].toString(); var re = new RegExp("0{"+imgStr.length+"}$"); var insideZipPrefix = this.subPrefix.match('[^/]+$'); var file = insideZipPrefix + '_' + this.imageFormat + '/' + insideZipPrefix + '_' + leafStr.replace(re, imgStr) + '.' + this.imageFormat; return file; } br.getPageSide = function(index) { //assume the book starts with a cover (right-hand leaf) //we should really get handside from scandata.xml // $$$ we should get this from scandata instead of assuming the accessible // leafs are contiguous if ('rl' != this.pageProgression) { // If pageProgression is not set RTL we assume it is LTR if (0 == (index & 0x1)) { // Even-numbered page return 'R'; } else { // Odd-numbered page return 'L'; } } else { // RTL if (0 == (index & 0x1)) { return 'L'; } else { return 'R'; } } } br.getPageNum = function(index) { var pageNum = this.pageNums[index]; if (pageNum) { return pageNum; } else { return 'n' + index; } } // Single images in the Internet Archive scandata.xml metadata are (somewhat incorrectly) // given a "leaf" number. Some of these images from the scanning process should not // be displayed in the BookReader (for example colour calibration cards). Since some // of the scanned images will not be displayed in the BookReader (those marked with // addToAccessFormats false in the scandata.xml) leaf numbers and BookReader page // indexes are generally not the same. This function returns the BookReader page // index given a scanned leaf number. // // This function is used, for example, to map between search results (that use the // leaf numbers) and the displayed pages in the BookReader. br.leafNumToIndex = function(leafNum) { for (var index = 0; index < this.leafMap.length; index++) { if (this.leafMap[index] == leafNum) { return index; } } return null; } // This function returns the left and right indices for the user-visible // spread that contains the given index. The return values may be // null if there is no facing page or the index is invalid. br.getSpreadIndices = function(pindex) { // $$$ we could make a separate function for the RTL case and // only bind it if necessary instead of always checking // $$$ we currently assume there are no gaps var spreadIndices = [null, null]; if ('rl' == this.pageProgression) { // Right to Left if (this.getPageSide(pindex) == 'R') { spreadIndices[1] = pindex; spreadIndices[0] = pindex + 1; } else { // Given index was LHS spreadIndices[0] = pindex; spreadIndices[1] = pindex - 1; } } else { // Left to right if (this.getPageSide(pindex) == 'L') { spreadIndices[0] = pindex; spreadIndices[1] = pindex + 1; } else { // Given index was RHS spreadIndices[1] = pindex; spreadIndices[0] = pindex - 1; } } //console.log(" index %d mapped to spread %d,%d", pindex, spreadIndices[0], spreadIndices[1]); return spreadIndices; } // Remove the page number assertions for all but the highest index page with // a given assertion. Ensures there is only a single page "{pagenum}" // e.g. the last page asserted as page 5 retains that assertion. br.uniquifyPageNums = function() { var seen = {}; for (var i = br.pageNums.length - 1; i--; i >= 0) { var pageNum = br.pageNums[i]; if ( !seen[pageNum] ) { seen[pageNum] = true; } else { br.pageNums[i] = null; } } } br.cleanupMetadata = function() { br.uniquifyPageNums(); } // getEmbedURL //________ // Returns a URL for an embedded version of the current book br.getEmbedURL = function(viewParams) { // We could generate a URL hash fragment here but for now we just leave at defaults var url = 'https://' + window.location.host + '/stream/'+this.bookId; if (this.subPrefix != this.bookId) { // Only include if needed url += '/' + this.subPrefix; } url += '?ui=embed'; if (typeof(viewParams) != 'undefined') { url += '#' + this.fragmentFromParams(viewParams); } return url; } // getEmbedCode //________ // Returns the embed code HTML fragment suitable for copy and paste br.getEmbedCode = function(frameWidth, frameHeight, viewParams) { return ""; } // getOpenLibraryRecord br.getOpenLibraryRecord = function(callback) { // Try looking up by ocaid first, then by source_record var self = this; // closure var jsonURL = self.olHost + '/query.json?type=/type/edition&*=&ocaid=' + self.bookId; $.ajax({ url: jsonURL, success: function(data) { if (data && data.length > 0) { callback(self, data[0]); } else { // try sourceid jsonURL = self.olHost + '/query.json?type=/type/edition&*=&source_records=ia:' + self.bookId; $.ajax({ url: jsonURL, success: function(data) { if (data && data.length > 0) { callback(self, data[0]); } }, dataType: 'jsonp' }); } }, dataType: 'jsonp' }); } br.buildInfoDiv = function(jInfoDiv) { // $$$ it might make more sense to have a URL on openlibrary.org that returns this info var escapedTitle = BookReader.util.escapeHTML(this.bookTitle); var domainRe = /(\w+\.(com|org))/; var domainMatch = domainRe.exec(this.bookUrl); var domain = this.bookUrl; if (domainMatch) { domain = domainMatch[1]; } // $$$ cover looks weird before it loads jInfoDiv.find('.BRfloatCover').append([ '
'].join('') ); var download_links = []; if (!this.olAuth) { download_links = [ 'More information on ' + domain + '
'); jInfoDiv.find('.BRfloatMeta').append(download_links.join('\n')); jInfoDiv.find('.BRfloatFoot').append([ '|', 'Report a problem', ].join('\n')); if (domain == 'archive.org') { jInfoDiv.find('.BRfloatMeta p.moreInfo span').css( {'background': 'url(https://archive.org/favicon.ico) no-repeat', 'width': 22, 'height': 18 } ); } jInfoDiv.find('.BRfloatTitle a').attr({'href': this.bookUrl, 'alt': this.bookTitle}).text(this.bookTitle); var bookPath = (window.location + '').replace('#','%23'); jInfoDiv.find('a.problem').attr('href','https://openlibrary.org/contact?path=' + bookPath); } br.pageW = [ 2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550,2550 ]; br.pageH = [ 3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301,3301 ]; br.leafMap = [ 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44 ]; br.pageNums = [ null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null,null ]; br.numLeafs = br.pageW.length; br.bookId = 'EinAstronomischerBeobachtungstextAusDem37.JahreNebukadnezarsIi'; br.zip = '/22/items/EinAstronomischerBeobachtungstextAusDem37.JahreNebukadnezarsIi/Nw1915OnVat4956-EnglishTranslationPdf_jp2.zip'; br.subPrefix = 'Nw1915OnVat4956-EnglishTranslationPdf'; br.server = 'ia600800.us.archive.org'; br.bookTitle= 'Ein astronomischer Beobachtungstext aus dem 37. Jahre Nebukadnezars II (-567/66), Paul V. Neugebauer and Ernst F. Weidner (1915), English translation (2011)'; br.bookPath = '/22/items/EinAstronomischerBeobachtungstextAusDem37.JahreNebukadnezarsIi/Nw1915OnVat4956-EnglishTranslationPdf'; br.bookUrl = 'https://archive.org/details/EinAstronomischerBeobachtungstextAusDem37.JahreNebukadnezarsIi'; br.imageFormat = 'jp2'; br.archiveFormat = 'zip'; br.pageProgression = 'lr'; br.olHost = 'https://openlibrary.org'; br.olAuthUrl = null; br.olAuth = false; // Check for config object // $$$ change this to use the newer params object if (typeof(brConfig) != 'undefined') { if (typeof(brConfig["ui"]) != 'undefined') { br.ui = brConfig["ui"]; } if (brConfig['mode'] == 1) { br.mode = 1; if (typeof(brConfig['reduce'] != 'undefined')) { br.reduce = brConfig['reduce']; } } else if (brConfig['mode'] == 2) { br.mode = 2; } if (typeof(brConfig["isAdmin"]) != 'undefined') { br.isAdmin = brConfig["isAdmin"]; } else { br.isAdmin = false; } } // brConfig function OLAuth() { this.olConnect = false; this.loanUUID = false; this.permsToken = false; var cookieRe = /;\s*/; var cookies = document.cookie.split(cookieRe); var length = cookies.length; var i; for (i=0; i', msg, '
', resolution, '
'].join('\n'); } OLAuth.prototype.showError = function(msg, resolution) { $(this.popup).css({ backgroundColor: "#fff", color: "#000" }); this.setPopupMsg(msg, resolution); } OLAuth.prototype.initCallback = function(obj) { if (false == obj.success) { if (br.isAdmin) { ret = confirm("We couldn't authenticate your loan with Open Library, but since you are an administrator or uploader of this book, you can access this book for QA purposes. Would you like to QA this book?"); if (!ret) { this.showError(obj.msg, obj.resolution) } else { br.init(); } } else { this.showError(obj.msg, obj.resolution) } } else { //user is authenticated this.setCookie(obj.token); this.olConnect = true; this.startPolling(); br.init(); } } OLAuth.prototype.callback = function(obj) { if (false == obj.success) { this.showPopup("#F0EEE2", "#000", obj.msg, obj.resolution); clearInterval(this.poller); this.ttsPoller = null; } else { this.olConnect = true; this.setCookie(obj.token); } } OLAuth.prototype.setCookie = function(value) { var date = new Date(); date.setTime(date.getTime()+(10*60*1000)); //10 min expiry var expiry = date.toGMTString(); var cookie = 'loan-'+br.bookId+'='+value; cookie += '; expires='+expiry; cookie += '; path=/; domain=.archive.org;'; document.cookie = cookie; this.permsToken = value; //refresh the br-loan uuid cookie with current expiry, if needed if (false !== this.loanUUID) { cookie = 'br-loan-'+br.bookId+'='+this.loanUUID; cookie += '; expires='+expiry; cookie += '; path=/; domain=.archive.org;'; document.cookie = cookie; } } OLAuth.prototype.deleteCookies = function() { var date = new Date(); date.setTime(date.getTime()-(24*60*60*1000)); //one day ago var expiry = date.toGMTString(); var cookie = 'loan-'+br.bookId+'=""'; cookie += '; expires='+expiry; cookie += '; path=/; domain=.archive.org;'; document.cookie = cookie; cookie = 'br-loan-'+br.bookId+'=""'; cookie += '; expires='+expiry; cookie += '; path=/; domain=.archive.org;'; document.cookie = cookie; } OLAuth.prototype.startPolling = function () { var self = this; this.poller=setInterval(function(){ if (!self.olConnect) { self.showPopup("#F0EEE2", "#000", 'Connection error', 'The BookReader cannot reach Open Library. This might mean that you are offline or that Open Library is down. Please check your Internet connection and refresh this page or try again later.'); clearInterval(self.poller); self.ttsPoller = null; } else { self.olConnect = false; self.callAuthUrl(); } },300000); //five minute interval } br.cleanupMetadata(); if (br.olAuth) { var olAuth = new OLAuth(); olAuth.init(); } else { br.init(); }