diff --git a/extensions/chrome/.gitignore b/extensions/chrome/.gitignore new file mode 100644 index 000000000..3eb92306c --- /dev/null +++ b/extensions/chrome/.gitignore @@ -0,0 +1 @@ +content/ diff --git a/extensions/firefox/.gitignore b/extensions/firefox/.gitignore index 08a23850c..6eec9a7f2 100644 --- a/extensions/firefox/.gitignore +++ b/extensions/firefox/.gitignore @@ -1,2 +1,4 @@ content/ metadata.inc +chrome.manifest.inc +locale/ diff --git a/extensions/firefox/chrome.manifest b/extensions/firefox/chrome.manifest new file mode 100644 index 000000000..97b76306b --- /dev/null +++ b/extensions/firefox/chrome.manifest @@ -0,0 +1,3 @@ +# Additional resources for pdf.js + +# PDFJS_SUPPORTED_LOCALES diff --git a/extensions/firefox/components/PdfStreamConverter.js b/extensions/firefox/components/PdfStreamConverter.js index 9f5cc6bee..ced6b1048 100644 --- a/extensions/firefox/components/PdfStreamConverter.js +++ b/extensions/firefox/components/PdfStreamConverter.js @@ -14,15 +14,26 @@ const PDF_CONTENT_TYPE = 'application/pdf'; const EXT_ID = 'uriloader@pdf.js'; const EXT_PREFIX = 'extensions.' + EXT_ID; const MAX_DATABASE_LENGTH = 4096; +const FIREFOX_ID = '{ec8030f7-c20a-464f-9b0e-13a3a9e97384}'; +const SEAMONKEY_ID = '{92650c4d-4b8e-4d2a-b7eb-24ecf4f6b63a}'; Cu.import('resource://gre/modules/XPCOMUtils.jsm'); Cu.import('resource://gre/modules/Services.jsm'); Cu.import('resource://gre/modules/NetUtil.jsm'); Cu.import('resource://gre/modules/AddonManager.jsm'); -let privateBrowsing = Cc['@mozilla.org/privatebrowsing;1'] - .getService(Ci.nsIPrivateBrowsingService); -let inPrivateBrowswing = privateBrowsing.privateBrowsingEnabled; +let appInfo = Cc['@mozilla.org/xre/app-info;1'] + .getService(Ci.nsIXULAppInfo); +let privateBrowsing, inPrivateBrowsing; + +if (appInfo.ID === FIREFOX_ID) { + privateBrowsing = Cc['@mozilla.org/privatebrowsing;1'] + .getService(Ci.nsIPrivateBrowsingService); + inPrivateBrowsing = privateBrowsing.privateBrowsingEnabled; +} else if (appInfo.ID === SEAMONKEY_ID) { + privateBrowsing = null; + inPrivateBrowsing = false; +} function getBoolPref(pref, def) { try { @@ -61,17 +72,37 @@ function getDOMWindow(aChannel) { return win; } -// Fake l10n until we get the real l10n sorted out. -var mozL10n = { - get: function(key, args, fallback) { - return fallback; +function getLocalizedStrings(path) { + var stringBundle = Cc['@mozilla.org/intl/stringbundle;1']. + getService(Ci.nsIStringBundleService). + createBundle('chrome://pdf.js/locale/' + path); + + var map = {}; + var enumerator = stringBundle.getSimpleEnumeration(); + while (enumerator.hasMoreElements()) { + var string = enumerator.getNext().QueryInterface(Ci.nsIPropertyElement); + var key = string.key, property = 'textContent'; + var i = key.lastIndexOf('.'); + if (i >= 0) { + property = key.substring(i + 1); + key = key.substring(0, i); + } + if (!(key in map)) + map[key] = {}; + map[key][property] = string.value; } -}; + return map; +} +function getLocalizedString(strings, id) { + if (id in strings) + return strings[id]['textContent']; + return id; +} // All the priviledged actions. function ChromeActions() { - this.inPrivateBrowswing = privateBrowsing.privateBrowsingEnabled; } + ChromeActions.prototype = { download: function(data) { let mimeService = Cc['@mozilla.org/mime;1'].getService(Ci.nsIMIMEService); @@ -107,7 +138,7 @@ ChromeActions.prototype = { channel.asyncOpen(listener, null); }, setDatabase: function(data) { - if (this.inPrivateBrowswing) + if (inPrivateBrowsing) return; // Protect against something sending tons of data to setDatabase. if (data.length > MAX_DATABASE_LENGTH) @@ -115,31 +146,43 @@ ChromeActions.prototype = { setStringPref(EXT_PREFIX + '.database', data); }, getDatabase: function() { - if (this.inPrivateBrowswing) + if (inPrivateBrowsing) return '{}'; return getStringPref(EXT_PREFIX + '.database', '{}'); }, getLocale: function() { return getStringPref('general.useragent.locale', 'en-US'); }, + getStrings: function(data) { + try { + // Lazy initialization of localizedStrings + if (!('localizedStrings' in this)) + this.localizedStrings = getLocalizedStrings('viewer.properties'); + + var result = this.localizedStrings[data]; + return JSON.stringify(result || null); + } catch (e) { + log('Unable to retrive localized strings: ' + e); + return 'null'; + } + }, pdfBugEnabled: function() { return getBoolPref(EXT_PREFIX + '.pdfBugEnabled', false); }, fallback: function(url) { + var strings = getLocalizedStrings('chrome.properties'); var self = this; - var message = mozL10n.get('unsupported_feature', null, - 'An unsupported feature was detected in this PDF document.'); + var message = getLocalizedString(strings, 'unsupported_feature'); var win = Services.wm.getMostRecentWindow('navigator:browser'); var notificationBox = win.gBrowser.getNotificationBox(); var buttons = [{ - label: mozL10n.get('download_document', null, 'Download Document'), + label: getLocalizedString(strings, 'download_document'), accessKey: null, callback: function() { self.download(url); } }, { - label: mozL10n.get('disable_viewer', null, - 'Disable Mozilla PDF Viewer'), + label: getLocalizedString(strings, 'disable_viewer'), accessKey: null, callback: function() { AddonManager.getAddonByID(EXT_ID, function(aAddon) { diff --git a/extensions/firefox/install.rdf b/extensions/firefox/install.rdf index d7eea9319..a5094c19a 100644 --- a/extensions/firefox/install.rdf +++ b/extensions/firefox/install.rdf @@ -8,11 +8,22 @@ PDF Viewer PDFJSSCRIPT_VERSION + + {ec8030f7-c20a-464f-9b0e-13a3a9e97384} 6.0 - 14.0a1 + 15.0a1 + + + + + + + {92650c4d-4b8e-4d2a-b7eb-24ecf4f6b63a} + 2.1.* + 2.12a1 true diff --git a/extensions/firefox/tools/l10n.js b/extensions/firefox/tools/l10n.js new file mode 100644 index 000000000..b16636e31 --- /dev/null +++ b/extensions/firefox/tools/l10n.js @@ -0,0 +1,114 @@ +/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */ + +'use strict'; + +// Small subset of the webL10n API by Fabien Cazenave for pdf.js extension. +(function(window) { + var gLanguage = ''; + + // fetch an l10n objects + function getL10nData(key) { + var response = FirefoxCom.request('getStrings', key); + var data = JSON.parse(response); + if (!data) + console.warn('[l10n] #' + key + ' missing for [' + gLanguage + ']'); + return data; + } + + // replace {{arguments}} with their values + function substArguments(text, args) { + if (!args) + return text; + + return text.replace(/\{\{\s*(\w+)\s*\}\}/g, function(all, name) { + return name in args ? args[name] : '{{' + name + '}}'; + }); + } + + // translate a string + function translateString(key, args, fallback) { + var data = getL10nData(key); + if (!data && fallback) + data = {textContent: fallback}; + if (!data) + return '{{' + key + '}}'; + return substArguments(data.textContent, args); + } + + // translate an HTML element + function translateElement(element) { + if (!element || !element.dataset) + return; + + // get the related l10n object + var data = getL10nData(element.dataset.l10nId); + if (!data) + return; + + // get arguments (if any) + // TODO: more flexible parser? + var args; + if (element.dataset.l10nArgs) try { + args = JSON.parse(element.dataset.l10nArgs); + } catch (e) { + console.warn('[l10n] could not parse arguments for #' + key + ''); + } + + // translate element + // TODO: security check? + for (var k in data) + element[k] = substArguments(data[k], args); + } + + + // translate an HTML subtree + function translateFragment(element) { + element = element || document.querySelector('html'); + + // check all translatable children (= w/ a `data-l10n-id' attribute) + var children = element.querySelectorAll('*[data-l10n-id]'); + var elementCount = children.length; + for (var i = 0; i < elementCount; i++) + translateElement(children[i]); + + // translate element itself if necessary + if (element.dataset.l10nId) + translateElement(element); + } + + window.addEventListener('DOMContentLoaded', function() { + gLanguage = FirefoxCom.request('getLocale', null); + + translateFragment(); + + // fire a 'localized' DOM event + var evtObject = document.createEvent('Event'); + evtObject.initEvent('localized', false, false); + evtObject.language = gLanguage; + window.dispatchEvent(evtObject); + }); + + // Public API + document.mozL10n = { + // get a localized string + get: translateString, + + // get|set the document language and direction + get language() { + return { + // get|set the document language (ISO-639-1) + get code() { return gLanguage; }, + + // get the direction (ltr|rtl) of the current language + get direction() { + // http://www.w3.org/International/questions/qa-scripts + // Arabic, Hebrew, Farsi, Pashto, Urdu + var rtlList = ['ar', 'he', 'fa', 'ps', 'ur']; + return (rtlList.indexOf(gLanguage) >= 0) ? 'rtl' : 'ltr'; + } + }; + } + }; +})(this); + diff --git a/l10n/en-US/chrome.properties b/l10n/en-US/chrome.properties new file mode 100644 index 000000000..f0f4bcb88 --- /dev/null +++ b/l10n/en-US/chrome.properties @@ -0,0 +1,3 @@ +unsupported_feature=An unsupported feature was detected in this PDF document. +download_document=Download Document +disable_viewer=Disable Mozilla PDF Viewer diff --git a/l10n/en-US/viewer.properties b/l10n/en-US/viewer.properties index c1617b0bd..c8dbe4aba 100644 --- a/l10n/en-US/viewer.properties +++ b/l10n/en-US/viewer.properties @@ -42,6 +42,3 @@ zoom_in_label=Zoom In zoom.title=Zoom thumb_page_title=Page {{page}} thumb_page_canvas=Thumbnail of Page {{page}} -unsupported_feature=An unsupported feature was detected in this PDF document. -download_document=Download Document -disable_pdfjs=Disable Mozilla PDF Viewer diff --git a/make.js b/make.js index 0a39e0b07..e166b6b96 100755 --- a/make.js +++ b/make.js @@ -6,6 +6,7 @@ var ROOT_DIR = __dirname + '/', // absolute path to project's root BUILD_TARGET = BUILD_DIR + 'pdf.js', FIREFOX_BUILD_DIR = BUILD_DIR + '/firefox/', EXTENSION_SRC_DIR = 'extensions/', + LOCALE_SRC_DIR = 'l10n/', GH_PAGES_DIR = BUILD_DIR + 'gh-pages/', REPO = 'git@github.com:mozilla/pdf.js.git', PYTHON_BIN = 'python2.7'; @@ -67,8 +68,9 @@ target.web = function() { // Creates localized resources for the viewer and extension. // target.locale = function() { - var L10N_PATH = 'l10n'; var METADATA_OUTPUT = 'extensions/firefox/metadata.inc'; + var CHROME_MANIFEST_OUTPUT = 'extensions/firefox/chrome.manifest.inc'; + var EXTENSION_LOCALE_OUTPUT = 'extensions/firefox/locale'; var VIEWER_OUTPUT = 'web/locale.properties'; var DEFAULT_LOCALE = 'en-US'; @@ -76,13 +78,17 @@ target.locale = function() { echo(); echo('### Building localization files'); - var subfolders = ls(L10N_PATH); + rm('-rf', EXTENSION_LOCALE_OUTPUT); + mkdir('-p', EXTENSION_LOCALE_OUTPUT); + + var subfolders = ls(LOCALE_SRC_DIR); subfolders.sort(); var metadataContent = ''; + var chromeManifestContent = ''; var viewerOutput = ''; for (var i = 0; i < subfolders.length; i++) { var locale = subfolders[i]; - var path = L10N_PATH + '/' + locale; + var path = LOCALE_SRC_DIR + locale; if (!test('-d', path)) continue; @@ -91,12 +97,17 @@ target.locale = function() { continue; } + mkdir('-p', EXTENSION_LOCALE_OUTPUT + '/' + locale); + chromeManifestContent += 'locale pdf.js ' + locale + ' locale/' + locale + '/\n'; + if (test('-f', path + '/viewer.properties')) { var properties = cat(path + '/viewer.properties'); - if (locale == DEFAULT_LOCALE) - viewerOutput = '[*]\n' + properties + '\n' + viewerOutput; - else - viewerOutput = viewerOutput + '[' + locale + ']\n' + properties + '\n'; + viewerOutput += '[' + locale + ']\n' + properties + '\n'; + cp(path + '/viewer.properties', EXTENSION_LOCALE_OUTPUT + '/' + locale); + } + + if (test('-f', path + '/chrome.properties')) { + cp(path + '/chrome.properties', EXTENSION_LOCALE_OUTPUT + '/' + locale); } if (test('-f', path + '/metadata.inc')) { @@ -106,6 +117,7 @@ target.locale = function() { } viewerOutput.to(VIEWER_OUTPUT); metadataContent.to(METADATA_OUTPUT); + chromeManifestContent.to(CHROME_MANIFEST_OUTPUT); }; // @@ -227,7 +239,7 @@ var EXTENSION_WEB_FILES = 'web/viewer.css', 'web/viewer.js', 'web/viewer.html', - 'external/webL10n/l10n.js', + 'extensions/firefox/tools/l10n.js', 'web/viewer-production.html'], EXTENSION_BASE_VERSION = 'f0f0418a9c6637981fe1182b9212c2d592774c7d', EXTENSION_VERSION_PREFIX = '0.3.', @@ -242,6 +254,7 @@ target.extension = function() { echo(); echo('### Building extensions'); + target.locale(); target.production(); target.firefox(); target.chrome(); @@ -277,21 +290,24 @@ target.firefox = function() { '*.rdf', '*.svg', '*.png', + '*.manifest', 'components', + 'locale', '../../LICENSE'], FIREFOX_EXTENSION_FILES = ['bootstrap.js', 'install.rdf', + 'chrome.manifest', 'icon.png', 'icon64.png', 'components', 'content', + 'locale', 'LICENSE'], FIREFOX_EXTENSION_NAME = 'pdf.js.xpi', FIREFOX_AMO_EXTENSION_NAME = 'pdf.js.amo.xpi'; - var LOCALE_CONTENT = cat('web/locale.properties'); - + target.locale(); target.production(); target.buildnumber(); cd(ROOT_DIR); @@ -310,7 +326,6 @@ target.firefox = function() { // Copy a standalone version of pdf.js inside the content directory cp(BUILD_TARGET, FIREFOX_BUILD_CONTENT_DIR + BUILD_DIR); cp('-R', EXTENSION_WEB_FILES, FIREFOX_BUILD_CONTENT_DIR + '/web'); - cp('web/locale.properties', FIREFOX_BUILD_CONTENT_DIR + '/web'); rm(FIREFOX_BUILD_CONTENT_DIR + '/web/viewer-production.html'); // Copy over the firefox extension snippet so we can inline pdf.js in it @@ -319,7 +334,6 @@ target.firefox = function() { // Modify the viewer so it does all the extension-only stuff. cd(FIREFOX_BUILD_CONTENT_DIR + '/web'); sed('-i', /.*PDFJSSCRIPT_INCLUDE_BUNDLE.*\n/, cat(ROOT_DIR + BUILD_TARGET), 'viewer-snippet-firefox-extension.html'); - sed('-i', /.*PDFJSSCRIPT_LOCALE_DATA.*\n/, LOCALE_CONTENT, 'viewer-snippet-firefox-extension.html'); sed('-i', /.*PDFJSSCRIPT_REMOVE_CORE.*\n/g, '', 'viewer.html'); sed('-i', /.*PDFJSSCRIPT_REMOVE_FIREFOX_EXTENSION.*\n/g, '', 'viewer.html'); sed('-i', /.*PDFJSSCRIPT_INCLUDE_FIREFOX_EXTENSION.*\n/, cat('viewer-snippet-firefox-extension.html'), 'viewer.html'); @@ -328,7 +342,6 @@ target.firefox = function() { // We don't need pdf.js anymore since its inlined rm('-Rf', FIREFOX_BUILD_CONTENT_DIR + BUILD_DIR); rm(FIREFOX_BUILD_CONTENT_DIR + '/web/viewer-snippet-firefox-extension.html'); - rm(FIREFOX_BUILD_CONTENT_DIR + '/web/locale.properties'); // Remove '.DS_Store' and other hidden files find(FIREFOX_BUILD_DIR).forEach(function(file) { if (file.match(/^\./)) @@ -342,6 +355,8 @@ target.firefox = function() { // Update localized metadata var localizedMetadata = cat(EXTENSION_SRC_DIR + '/firefox/metadata.inc'); sed('-i', /.*PDFJS_LOCALIZED_METADATA.*\n/, localizedMetadata, FIREFOX_BUILD_DIR + '/install.rdf'); + var chromeManifest = cat(EXTENSION_SRC_DIR + '/firefox/chrome.manifest.inc'); + sed('-i', /.*PDFJS_SUPPORTED_LOCALES.*\n/, chromeManifest, FIREFOX_BUILD_DIR + '/chrome.manifest'); // Create the xpi cd(FIREFOX_BUILD_DIR); @@ -365,25 +380,27 @@ target.mozcentral = function() { echo(); echo('### Building mozilla-central extension'); - var MOZCENTRAL_DIR = BUILD_DIR + '/mozcentral', - MOZCENTRAL_CONTENT_DIR = MOZCENTRAL_DIR + '/content/', - MOZCENTRAL_L10N_DIR = MOZCENTRAL_DIR + '/l10n/', + var MOZCENTRAL_DIR = BUILD_DIR + 'mozcentral/', + MOZCENTRAL_EXTENSION_DIR = MOZCENTRAL_DIR + 'browser/app/profile/extensions/uriloader@pdf.js/', + MOZCENTRAL_CONTENT_DIR = MOZCENTRAL_EXTENSION_DIR + 'content/', + MOZCENTRAL_L10N_DIR = MOZCENTRAL_DIR + 'browser/locales/en-US/pdfviewer/', FIREFOX_CONTENT_DIR = EXTENSION_SRC_DIR + '/firefox/content/', FIREFOX_EXTENSION_FILES_TO_COPY = ['*.js', '*.svg', '*.png', + '*.manifest', 'install.rdf.in', 'README.mozilla', 'components', '../../LICENSE'], DEFAULT_LOCALE_FILES = - ['l10n/en-US/viewer.properties', - 'l10n/en-US/metadata.inc'], + [LOCALE_SRC_DIR + 'en-US/viewer.properties'], FIREFOX_MC_EXTENSION_FILES = ['bootstrap.js', 'icon.png', 'icon64.png', + 'chrome.manifest', 'components', 'content', 'LICENSE']; @@ -401,7 +418,7 @@ target.mozcentral = function() { // Copy extension files cd('extensions/firefox'); - cp('-R', FIREFOX_EXTENSION_FILES_TO_COPY, ROOT_DIR + MOZCENTRAL_DIR); + cp('-R', FIREFOX_EXTENSION_FILES_TO_COPY, ROOT_DIR + MOZCENTRAL_EXTENSION_DIR); cd(ROOT_DIR); // Copy a standalone version of pdf.js inside the content directory @@ -433,11 +450,11 @@ target.mozcentral = function() { cp(DEFAULT_LOCALE_FILES, MOZCENTRAL_L10N_DIR); // Update the build version number - sed('-i', /PDFJSSCRIPT_VERSION/, EXTENSION_VERSION, MOZCENTRAL_DIR + '/install.rdf.in'); - sed('-i', /PDFJSSCRIPT_VERSION/, EXTENSION_VERSION, MOZCENTRAL_DIR + '/README.mozilla'); + sed('-i', /PDFJSSCRIPT_VERSION/, EXTENSION_VERSION, MOZCENTRAL_EXTENSION_DIR + 'install.rdf.in'); + sed('-i', /PDFJSSCRIPT_VERSION/, EXTENSION_VERSION, MOZCENTRAL_EXTENSION_DIR + 'README.mozilla'); // List all files for mozilla-central - cd(MOZCENTRAL_DIR); + cd(MOZCENTRAL_EXTENSION_DIR); var extensionFiles = ''; find(FIREFOX_MC_EXTENSION_FILES).forEach(function(file){ if (test('-f', file)) diff --git a/src/api.js b/src/api.js index 0e76a17ae..c5600b57e 100644 --- a/src/api.js +++ b/src/api.js @@ -8,10 +8,12 @@ * e.g. No cross domain requests without CORS. * * @param {string|TypedAray} source Either a url to a PDF is located or a - * typed array already populated with data. + * typed array (Uint8Array) already populated with data. + * @param {Object} headers An object containing the http headers like this: + * { Authorization: "BASIC XXX" }. * @return {Promise} A promise that is resolved with {PDFDocumentProxy} object. */ -PDFJS.getDocument = function getDocument(source) { +PDFJS.getDocument = function getDocument(source, headers) { var promise = new PDFJS.Promise(); var transport = new WorkerTransport(promise); if (typeof source === 'string') { @@ -29,7 +31,8 @@ PDFJS.getDocument = function getDocument(source) { error: function getPDFError(e) { promise.reject('Unexpected server response of ' + e.target.status + '.'); - } + }, + headers: headers }, function getPDFLoad(data) { transport.sendData(data); diff --git a/src/core.js b/src/core.js index 13e14c702..48812c3d2 100644 --- a/src/core.js +++ b/src/core.js @@ -31,7 +31,19 @@ function getPdf(arg, callback) { params = { url: arg }; var xhr = new XMLHttpRequest(); + xhr.open('GET', params.url); + + var headers = params.headers; + if (headers) { + for (var property in headers) { + if (typeof headers[property] === 'undefined') + continue; + + xhr.setRequestHeader(property, params.headers[property]); + } + } + xhr.mozResponseType = xhr.responseType = 'arraybuffer'; var protocol = params.url.indexOf(':') < 0 ? window.location.protocol : params.url.substring(0, params.url.indexOf(':') + 1); diff --git a/src/evaluator.js b/src/evaluator.js index d2ca8cfdb..b0408998e 100644 --- a/src/evaluator.js +++ b/src/evaluator.js @@ -112,14 +112,33 @@ var PartialEvaluator = (function PartialEvaluatorClosure() { }; function splitCombinedOperations(operations) { - // Two operations can be combined together, trying to find which two + // Two or more operations can be combined together, trying to find which // operations were concatenated. - for (var i = operations.length - 1; i > 0; i--) { - var op1 = operations.substring(0, i), op2 = operations.substring(i); - if (op1 in OP_MAP && op2 in OP_MAP) - return [op1, op2]; // operations found + var result = []; + var opIndex = 0; + + if (!operations) { + return null; } - return null; + + while (opIndex < operations.length) { + var currentOp = ''; + for (var op in OP_MAP) { + if (op == operations.substr(opIndex, op.length) && + op.length > currentOp.length) { + currentOp = op; + } + } + + if (currentOp.length > 0) { + result.push(operations.substr(opIndex, currentOp.length)); + opIndex += currentOp.length; + } else { + return null; + } + } + + return result; } PartialEvaluator.prototype = { @@ -267,14 +286,14 @@ var PartialEvaluator = (function PartialEvaluatorClosure() { var patterns = resources.get('Pattern') || new Dict(); var parser = new Parser(new Lexer(stream), false, xref); var res = resources; - var hasNextObj = false, nextObj; + var hasNextObj = false, nextObjs; var args = [], obj; var TILING_PATTERN = 1, SHADING_PATTERN = 2; while (true) { if (hasNextObj) { - obj = nextObj; - hasNextObj = false; + obj = nextObjs.pop(); + hasNextObj = (nextObjs.length > 0); } else { obj = parser.getObj(); if (isEOF(obj)) @@ -290,9 +309,12 @@ var PartialEvaluator = (function PartialEvaluatorClosure() { if (cmds) { cmd = cmds[0]; fn = OP_MAP[cmd]; - // feeding other command on the next interation + // feeding other command on the next iteration hasNextObj = true; - nextObj = Cmd.get(cmds[1]); + nextObjs = []; + for (var idx = 1; idx < cmds.length; idx++) { + nextObjs.push(Cmd.get(cmds[idx])); + } } } assertWellFormed(fn, 'Unknown command "' + cmd + '"'); diff --git a/test/unit/evaluator_spec.js b/test/unit/evaluator_spec.js new file mode 100644 index 000000000..4ee0768a7 --- /dev/null +++ b/test/unit/evaluator_spec.js @@ -0,0 +1,83 @@ +/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */ + +'use strict'; + +describe('evaluator', function() { + function XrefMock(queue) { + this.queue = queue; + } + XrefMock.prototype = { + fetchIfRef: function() { + return this.queue.shift(); + } + }; + function HandlerMock() { + this.inputs = []; + } + HandlerMock.prototype = { + send: function(name, data) { + this.inputs({name: name, data: data}); + } + }; + function ResourcesMock() { } + ResourcesMock.prototype = { + get: function(name) { + return this[name]; + } + }; + + describe('splitCombinedOperations', function() { + it('should reject unknown operations', function() { + var evaluator = new PartialEvaluator(new XrefMock(), new HandlerMock(), + 'prefix'); + var stream = new StringStream('qTT'); + var thrown = false; + try { + evaluator.getOperatorList(stream, new ResourcesMock(), []); + } catch (e) { + thrown = e; + } + expect(thrown).toNotEqual(false); + }); + + it('should handle one operations', function() { + var evaluator = new PartialEvaluator(new XrefMock(), new HandlerMock(), + 'prefix'); + var stream = new StringStream('Q'); + var result = evaluator.getOperatorList(stream, new ResourcesMock(), []); + + expect(!!result.fnArray && !!result.argsArray).toEqual(true); + expect(result.fnArray.length).toEqual(1); + expect(result.fnArray[0]).toEqual('restore'); + }); + + it('should handle two glued operations', function() { + var evaluator = new PartialEvaluator(new XrefMock(), new HandlerMock(), + 'prefix'); + var resources = new ResourcesMock(); + resources.Res1 = {}; + var stream = new StringStream('/Res1 DoQ'); + var result = evaluator.getOperatorList(stream, resources, []); + + expect(!!result.fnArray && !!result.argsArray).toEqual(true); + expect(result.fnArray.length).toEqual(2); + expect(result.fnArray[0]).toEqual('paintXObject'); + expect(result.fnArray[1]).toEqual('restore'); + }); + + it('should handle tree glued operations', function() { + var evaluator = new PartialEvaluator(new XrefMock(), new HandlerMock(), + 'prefix'); + var stream = new StringStream('qqq'); + var result = evaluator.getOperatorList(stream, new ResourcesMock(), []); + + expect(!!result.fnArray && !!result.argsArray).toEqual(true); + expect(result.fnArray.length).toEqual(3); + expect(result.fnArray[0]).toEqual('save'); + expect(result.fnArray[1]).toEqual('save'); + expect(result.fnArray[2]).toEqual('save'); + }); + }); +}); + diff --git a/test/unit/unit_test.html b/test/unit/unit_test.html index cdd0c297f..ca0a1aed1 100644 --- a/test/unit/unit_test.html +++ b/test/unit/unit_test.html @@ -39,6 +39,7 @@ + diff --git a/web/viewer.js b/web/viewer.js index 34118e2aa..0a7594b35 100644 --- a/web/viewer.js +++ b/web/viewer.js @@ -1378,11 +1378,12 @@ window.addEventListener('load', function webViewerLoad(evt) { if ('disableWorker' in hashParams) PDFJS.disableWorker = (hashParams['disableWorker'] === 'true'); - var locale = !PDFJS.isFirefoxExtension ? navigator.language : - FirefoxCom.request('getLocale', null); - if ('locale' in hashParams) - locale = hashParams['locale']; - mozL10n.language.code = locale; + if (!PDFJS.isFirefoxExtension) { + var locale = navigator.language; + if ('locale' in hashParams) + locale = hashParams['locale']; + mozL10n.language.code = locale; + } if ('disableTextLayer' in hashParams) PDFJS.disableTextLayer = (hashParams['disableTextLayer'] === 'true');