UNPKG

objj-runtime

Version:

JavaScript (ECMAScript) and Objective-J runtime

919 lines (701 loc) 26.1 kB
/* * CFBundle.js * Objective-J * * Created by Francisco Tolmasky. * Copyright 2008-2010, 280 North, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ var CFBundleUnloaded = 0, CFBundleLoading = 1 << 0, CFBundleLoadingInfoPlist = 1 << 1, CFBundleLoadingExecutable = 1 << 2, CFBundleLoadingSpritedImages = 1 << 3, CFBundleLoadingLocalizableStrings = 1 << 4, CFBundleLoaded = 1 << 5; var CFBundlesForURLStrings = { }, CFBundlesForClasses = { }, CFBundlesWithIdentifiers = { }, CFCacheBuster = new Date().getTime(), CFTotalBytesLoaded = 0, CPApplicationSizeInBytes = 0; var CPBundleDefaultBrowserLanguage = "CPBundleDefaultBrowserLanguage", CPBundleDefaultLanguage = "CPBundleDefaultLanguage"; GLOBAL(CFBundle) = function(/*CFURL|String*/ aURL) { aURL = makeAbsoluteURL(aURL).asDirectoryPathURL(); var URLString = aURL.absoluteString(), existingBundle = CFBundlesForURLStrings[URLString]; if (existingBundle) return existingBundle; CFBundlesForURLStrings[URLString] = this; this._bundleURL = aURL; this._resourcesDirectoryURL = new CFURL("Resources/", aURL); this._staticResource = NULL; this._isValid = NO; this._loadStatus = CFBundleUnloaded; this._loadRequests = []; this._infoDictionary = new CFDictionary(); this._eventDispatcher = new EventDispatcher(this); this._localizableStrings = []; this._loadedLanguage = NULL; } DISPLAY_NAME(CFBundle); CFBundle.environments = function() { // Passed in by GCC. return ENVIRONMENTS; }; DISPLAY_NAME(CFBundle.environments); CFBundle.bundleContainingURL = function(/*CFURL|String*/ aURL) { aURL = new CFURL(".", makeAbsoluteURL(aURL)); var previousURLString, URLString = aURL.absoluteString(); while (!previousURLString || previousURLString !== URLString) { var bundle = CFBundlesForURLStrings[URLString]; if (bundle && bundle._isValid) return bundle; aURL = new CFURL("..", aURL); previousURLString = URLString; URLString = aURL.absoluteString(); } return NULL; }; DISPLAY_NAME(CFBundle.bundleContainingURL); CFBundle.mainBundle = function() { return new CFBundle(mainBundleURL); }; DISPLAY_NAME(CFBundle.mainBundle); function addClassToBundle(aClass, aBundle) { if (aBundle) CFBundlesForClasses[aClass.name] = aBundle; } function resetBundle() { CFBundlesForURLStrings = { }; CFBundlesForClasses = { }; CFBundlesWithIdentifiers = { }; //CFCacheBuster = new Date().getTime(), CFTotalBytesLoaded = 0; CPApplicationSizeInBytes = 0; } CFBundle.bundleForClass = function(/*Class*/ aClass) { return CFBundlesForClasses[aClass.name] || CFBundle.mainBundle(); }; DISPLAY_NAME(CFBundle.bundleForClass); CFBundle.bundleWithIdentifier = function(/*String*/ bundleID) { return CFBundlesWithIdentifiers[bundleID] || NULL; }; DISPLAY_NAME(CFBundle.bundleWithIdentifier); CFBundle.prototype.bundleURL = function() { return this._bundleURL.absoluteURL(); }; DISPLAY_NAME(CFBundle.prototype.bundleURL); CFBundle.prototype.resourcesDirectoryURL = function() { return this._resourcesDirectoryURL; }; DISPLAY_NAME(CFBundle.prototype.resourcesDirectoryURL); CFBundle.prototype.resourceURL = function(/*String*/ aResourceName, /*String*/ aType, /*String*/ aSubDirectory, /*String*/ localizationName) { if (aType) aResourceName = aResourceName + "." + aType; if (localizationName) aResourceName = localizationName + aResourceName; if (aSubDirectory) aResourceName = aSubDirectory + "/" + aResourceName; var resourceURL = (new CFURL(aResourceName, this.resourcesDirectoryURL())).mappedURL(); return resourceURL.absoluteURL(); }; DISPLAY_NAME(CFBundle.prototype.resourceURL); CFBundle.prototype.mostEligibleEnvironmentURL = function() { if (this._mostEligibleEnvironmentURL === undefined) this._mostEligibleEnvironmentURL = new CFURL(this.mostEligibleEnvironment() + ".environment/", this.bundleURL()); return this._mostEligibleEnvironmentURL; } DISPLAY_NAME(CFBundle.prototype.mostEligibleEnvironmentURL); CFBundle.prototype.executableURL = function() { if (this._executableURL === undefined) { var executableSubPath = this.valueForInfoDictionaryKey("CPBundleExecutable"); if (!executableSubPath) this._executableURL = NULL; else this._executableURL = new CFURL(executableSubPath, this.mostEligibleEnvironmentURL()); } return this._executableURL; }; DISPLAY_NAME(CFBundle.prototype.executableURL); CFBundle.prototype.infoDictionary = function() { return this._infoDictionary; }; DISPLAY_NAME(CFBundle.prototype.infoDictionary); CFBundle.prototype.loadedLanguage = function() { return this._loadedLanguage; }; CFBundle.prototype.valueForInfoDictionaryKey = function(/*String*/ aKey) { return this._infoDictionary.valueForKey(aKey); }; DISPLAY_NAME(CFBundle.prototype.valueForInfoDictionaryKey); CFBundle.prototype.identifier = function() { return this._infoDictionary.valueForKey("CPBundleIdentifier"); }; DISPLAY_NAME(CFBundle.prototype.identifier); CFBundle.prototype.hasSpritedImages = function() { var environments = this._infoDictionary.valueForKey("CPBundleEnvironmentsWithImageSprites") || [], index = environments.length, mostEligibleEnvironment = this.mostEligibleEnvironment(); while (index--) if (environments[index] === mostEligibleEnvironment) return YES; return NO; }; DISPLAY_NAME(CFBundle.prototype.hasSpritedImages); CFBundle.prototype.environments = function() { return this._infoDictionary.valueForKey("CPBundleEnvironments") || ["ObjJ"]; }; DISPLAY_NAME(CFBundle.prototype.environments); CFBundle.prototype.mostEligibleEnvironment = function(/*Array*/ environments) { environments = environments || this.environments(); var objj_environments = CFBundle.environments(), index = 0, count = objj_environments.length, innerCount = environments.length; // Ugh, no indexOf, no objects-in-common. for(; index < count; ++index) { var innerIndex = 0, environment = objj_environments[index]; for (; innerIndex < innerCount; ++innerIndex) if(environment === environments[innerIndex]) return environment; } return NULL; }; DISPLAY_NAME(CFBundle.prototype.mostEligibleEnvironment); CFBundle.prototype.isLoading = function() { return this._loadStatus & CFBundleLoading; }; DISPLAY_NAME(CFBundle.prototype.isLoading); CFBundle.prototype.isLoaded = function() { return !!(this._loadStatus & CFBundleLoaded); }; DISPLAY_NAME(CFBundle.prototype.isLoaded); CFBundle.prototype.load = function(/*BOOL*/ shouldExecute) { if (this._loadStatus !== CFBundleUnloaded) return; this._loadStatus = CFBundleLoading | CFBundleLoadingInfoPlist; var self = this, bundleURL = this.bundleURL(), parentURL = new CFURL("..", bundleURL); if (parentURL.absoluteString() === bundleURL.absoluteString()) parentURL = parentURL.schemeAndAuthority(); StaticResource.resolveResourceAtURL(parentURL, YES, function(aStaticResource) { var resourceName = bundleURL.lastPathComponent(); self._staticResource = aStaticResource._children[resourceName] || new StaticResource(bundleURL, aStaticResource, YES, NO); function onsuccess(/*Event*/ anEvent) { self._loadStatus &= ~CFBundleLoadingInfoPlist; var infoDictionary = anEvent.request.responsePropertyList(); self._isValid = !!infoDictionary || CFBundle.mainBundle() === self; if (infoDictionary) { self._infoDictionary = infoDictionary; var identifier = self._infoDictionary.valueForKey("CPBundleIdentifier"); if (identifier) CFBundlesWithIdentifiers[identifier] = self; } if (!self._infoDictionary) { finishBundleLoadingWithError(self, new Error("Could not load bundle at \"" + path + "\"")); return; } if (self === CFBundle.mainBundle() && self.valueForInfoDictionaryKey("CPApplicationSize")) CPApplicationSizeInBytes = self.valueForInfoDictionaryKey("CPApplicationSize").valueForKey("executable") || 0; loadLanguageForBundle(self); loadExecutableAndResources(self, shouldExecute); } function onfailure() { self._isValid = CFBundle.mainBundle() === self; self._loadStatus = CFBundleUnloaded; finishBundleLoadingWithError(self, new Error("Could not load bundle at \"" + self.bundleURL() + "\"")); } new FileRequest(new CFURL("Info.plist", self.bundleURL()), onsuccess, onfailure); }); }; DISPLAY_NAME(CFBundle.prototype.load); function finishBundleLoadingWithError(/*CFBundle*/ aBundle, /*Event*/ anError) { resolveStaticResource(aBundle._staticResource); aBundle._eventDispatcher.dispatchEvent( { type:"error", error:anError, bundle:aBundle }); } function loadExecutableAndResources(/*Bundle*/ aBundle, /*BOOL*/ shouldExecute) { if (!aBundle.mostEligibleEnvironment()) return failure(); loadExecutableForBundle(aBundle, success, failure, progress); loadSpritedImagesForBundle(aBundle, success, failure, progress); loadLocalizableStringsForBundle(aBundle, success, failure, progress); if (aBundle._loadStatus === CFBundleLoading) return success(); function failure(/*Error*/ anError) { var loadRequests = aBundle._loadRequests, count = loadRequests.length; while (count--) loadRequests[count].abort(); this._loadRequests = []; aBundle._loadStatus = CFBundleUnloaded; finishBundleLoadingWithError(aBundle, anError || new Error("Could not recognize executable code format in Bundle " + aBundle)); } function progress(bytesLoaded) { if ((typeof CPApp === "undefined" || !CPApp || !CPApp._finishedLaunching) && typeof OBJJ_PROGRESS_CALLBACK === "function") { CFTotalBytesLoaded += bytesLoaded; var percent = CPApplicationSizeInBytes ? MAX(MIN(1.0, CFTotalBytesLoaded / CPApplicationSizeInBytes), 0.0) : 0; OBJJ_PROGRESS_CALLBACK(percent, CPApplicationSizeInBytes, aBundle.bundlePath()); } } function success() { if (aBundle._loadStatus === CFBundleLoading) aBundle._loadStatus = CFBundleLoaded; else return; // Set resolved to true here in case during evaluation this bundle // needs to resolve another bundle which in turn needs it to be resolved (cycle). resolveStaticResource(aBundle._staticResource); function complete() { aBundle._eventDispatcher.dispatchEvent( { type:"load", bundle:aBundle }); } if (shouldExecute) executeBundle(aBundle, complete); else complete(); } } function loadExecutableForBundle(/*Bundle*/ aBundle, success, failure, progress) { var executableURL = aBundle.executableURL(); if (!executableURL) return; aBundle._loadStatus |= CFBundleLoadingExecutable; new FileRequest(executableURL, function(/*Event*/ anEvent) { try { decompileStaticFile(aBundle, anEvent.request.responseText(), executableURL); aBundle._loadStatus &= ~CFBundleLoadingExecutable; success(); } catch(anException) { failure(anException); } }, failure, progress); } function spritedImagesTestURLStringForBundle(/*Bundle*/ aBundle) { return "mhtml:" + new CFURL("MHTMLTest.txt", aBundle.mostEligibleEnvironmentURL()); } function spritedImagesURLForBundle(/*Bundle*/ aBundle) { if (CFBundleSupportedSpriteType === CFBundleDataURLSpriteType) return new CFURL("dataURLs.txt", aBundle.mostEligibleEnvironmentURL()); if (CFBundleSupportedSpriteType === CFBundleMHTMLSpriteType || CFBundleSupportedSpriteType === CFBundleMHTMLUncachedSpriteType) return new CFURL("MHTMLPaths.txt", aBundle.mostEligibleEnvironmentURL()); return NULL; } function loadSpritedImagesForBundle(/*Bundle*/ aBundle, success, failure, progress) { if (!aBundle.hasSpritedImages()) return; aBundle._loadStatus |= CFBundleLoadingSpritedImages; if (!CFBundleHasTestedSpriteSupport()) return CFBundleTestSpriteSupport(spritedImagesTestURLStringForBundle(aBundle), function() { loadSpritedImagesForBundle(aBundle, success, failure, progress); }); var spritedImagesURL = spritedImagesURLForBundle(aBundle); if (!spritedImagesURL) { aBundle._loadStatus &= ~CFBundleLoadingSpritedImages; return success(); } new FileRequest(spritedImagesURL, function(/*Event*/ anEvent) { try { decompileStaticFile(aBundle, anEvent.request.responseText(), spritedImagesURL); aBundle._loadStatus &= ~CFBundleLoadingSpritedImages; success(); } catch(anException) { failure(anException); } }, failure, progress); } function loadLocalizableStringsForBundle(/*Bundle*/ aBundle, success, failure, progress) { var language = aBundle._loadedLanguage; if (!language) return; var localizableStrings = aBundle.valueForInfoDictionaryKey("CPBundleLocalizableStrings"); if (!localizableStrings) return; var self = aBundle, length = localizableStrings.length, languagePathURL = new CFURL(language + ".lproj/", self.resourcesDirectoryURL()), fileSuccessed = 0; for (var i = 0; i < length; i++) { var localizableString = localizableStrings[i]; function onsuccess(/*Event*/ anEvent) { var contentFile = anEvent.request.responseText(), tableName = new CFURL(anEvent.request._URL).lastPathComponent(); try { loadLocalizableContentForFileInBundle(self, contentFile, tableName); if (++fileSuccessed == length) { aBundle._loadStatus &= ~CFBundleLoadingLocalizableStrings; success(); } } catch(e) { failure(new Error("Error when parsing the localizable file " + tableName)); } } aBundle._loadStatus |= CFBundleLoadingLocalizableStrings; new FileRequest(new CFURL(localizableString, languagePathURL), onsuccess, failure, progress); } } function loadLocalizableContentForFileInBundle(bundle, contentFile, tableName) { var values = {}, lines = contentFile.split("\n"), currentContext; bundle._localizableStrings[tableName] = values; for (var i = 0; i < lines.length; i++) { var line = lines[i]; // Context if (line[0] == "/") { currentContext = line.substring(2, line.length - 2).trim(); continue; } // Key if (line[0] == "\"") { var split = line.split("\"") // Here we add twice the translated value, once with the context and once without // This will help in term of rapidity when asking the value of the key var key = split[1]; if (!(key in values)) values[key] = split[3]; key += currentContext; if (!(key in values)) values[key] = split[3]; continue; } } } function loadLanguageForBundle(aBundle) { if (aBundle._loadedLanguage) return; var defaultLanguage = aBundle.valueForInfoDictionaryKey(CPBundleDefaultLanguage); if (defaultLanguage != CPBundleDefaultBrowserLanguage && defaultLanguage) { aBundle._loadedLanguage = defaultLanguage; return; } if (typeof navigator == "undefined") return; // userLanguage is an IE only property. var language = (typeof navigator.language !== "undefined") ? navigator.language : navigator.userLanguage; if (!language) return; aBundle._loadedLanguage = language.substring(0, 2); } var CFBundleSpriteSupportListeners = [], CFBundleSupportedSpriteType = -1, CFBundleNoSpriteType = 0, CFBundleDataURLSpriteType = 1, CFBundleMHTMLSpriteType = 2, CFBundleMHTMLUncachedSpriteType = 3; function CFBundleHasTestedSpriteSupport() { return CFBundleSupportedSpriteType !== -1; } function CFBundleTestSpriteSupport(/*String*/ MHTMLPath, /*Function*/ aCallback) { if (CFBundleHasTestedSpriteSupport()) return; CFBundleSpriteSupportListeners.push(aCallback); if (CFBundleSpriteSupportListeners.length > 1) return; CFBundleSpriteSupportListeners.push(function() { var size = 0, sizeDictionary = CFBundle.mainBundle().valueForInfoDictionaryKey("CPApplicationSize"); if (!sizeDictionary) return; switch (CFBundleSupportedSpriteType) { case CFBundleDataURLSpriteType: size = sizeDictionary.valueForKey("data"); break; case CFBundleMHTMLSpriteType: case CFBundleMHTMLUncachedSpriteType: size = sizeDictionary.valueForKey("mhtml"); break; } CPApplicationSizeInBytes += size; }); CFBundleTestSpriteTypes([ CFBundleDataURLSpriteType, "", CFBundleMHTMLSpriteType, MHTMLPath+"!test", CFBundleMHTMLUncachedSpriteType, MHTMLPath+"?"+CFCacheBuster+"!test" ]); } function CFBundleNotifySpriteSupportListeners() { var count = CFBundleSpriteSupportListeners.length; while (count--) CFBundleSpriteSupportListeners[count](); } function CFBundleTestSpriteTypes(/*Array*/ spriteTypes) { // If we don't support Images, then clearly we don't support sprites. if (!("Image" in global) || spriteTypes.length < 2) { CFBundleSupportedSpriteType = CFBundleNoSpriteType; CFBundleNotifySpriteSupportListeners(); return; } var image = new Image(); image.onload = function() { if (image.width === 1 && image.height === 1) { CFBundleSupportedSpriteType = spriteTypes[0]; CFBundleNotifySpriteSupportListeners(); } else image.onerror(); }; image.onerror = function() { CFBundleTestSpriteTypes(spriteTypes.slice(2)); }; image.src = spriteTypes[1]; } function executeBundle(/*Bundle*/ aBundle, /*Function*/ aCallback) { var staticResources = [aBundle._staticResource]; function executeStaticResources(index) { for (; index < staticResources.length; ++index) { var staticResource = staticResources[index]; if (staticResource.isNotFound()) continue; if (staticResource.isFile()) { var executable = new FileExecutable(staticResource.URL()); if (executable.hasLoadedFileDependencies()) executable.execute(); else { executable.loadFileDependencies(function() { executeStaticResources(index); }); return; } } else //if (staticResource.isDirectory()) { // We don't want to execute resources. if (staticResource.URL().absoluteString() === aBundle.resourcesDirectoryURL().absoluteString()) continue; var children = staticResource.children(); for (var name in children) if (hasOwnProperty.call(children, name)) staticResources.push(children[name]); } } aCallback(); } executeStaticResources(0); } var STATIC_MAGIC_NUMBER = "@STATIC", MARKER_PATH = "p", MARKER_URI = "u", MARKER_CODE = "c", MARKER_TEXT = "t", MARKER_IMPORT_STD = 'I', MARKER_IMPORT_LOCAL = 'i'; MARKER_SOURCE_MAP = 'S'; function decompileStaticFile(/*Bundle*/ aBundle, /*String*/ aString, /*String*/ aPath) { var stream = new MarkedStream(aString); if (stream.magicNumber() !== STATIC_MAGIC_NUMBER) throw new Error("Could not read static file: " + aPath); if (stream.version() !== "1.0") throw new Error("Could not read static file: " + aPath); var marker, bundleURL = aBundle.bundleURL(), file = NULL; while (marker = stream.getMarker()) { var text = stream.getString(); if (marker === MARKER_PATH) { var fileURL = new CFURL(text, bundleURL), parent = StaticResource.resourceAtURL(new CFURL(".", fileURL), YES); file = new StaticResource(fileURL, parent, NO, YES); } else if (marker === MARKER_URI) { var URL = new CFURL(text, bundleURL), mappedURLString = stream.getString(); if (mappedURLString.indexOf("mhtml:") === 0) { mappedURLString = "mhtml:" + new CFURL(mappedURLString.substr("mhtml:".length), bundleURL); if (CFBundleSupportedSpriteType === CFBundleMHTMLUncachedSpriteType) { var exclamationIndex = mappedURLString.indexOf("!"), firstPart = mappedURLString.substring(0, exclamationIndex), lastPart = mappedURLString.substring(exclamationIndex); mappedURLString = firstPart + "?" + CFCacheBuster + lastPart; } } CFURL.setMappedURLForURL(URL, new CFURL(mappedURLString)); // The unresolved directories must not be bundles. var parent = StaticResource.resourceAtURL(new CFURL(".", URL), YES); new StaticResource(URL, parent, NO, YES); } else if (marker === MARKER_TEXT) file.write(text); } } // Event Managament CFBundle.prototype.addEventListener = function(/*String*/ anEventName, /*Function*/ anEventListener) { this._eventDispatcher.addEventListener(anEventName, anEventListener); }; DISPLAY_NAME(CFBundle.prototype.addEventListener); CFBundle.prototype.removeEventListener = function(/*String*/ anEventName, /*Function*/ anEventListener) { this._eventDispatcher.removeEventListener(anEventName, anEventListener); }; DISPLAY_NAME(CFBundle.prototype.removeEventListener); CFBundle.prototype.onerror = function(/*Event*/ anEvent) { throw anEvent.error; }; DISPLAY_NAME(CFBundle.prototype.onerror); CFBundle.prototype.bundlePath = function() { return this.bundleURL().path(); }; CFBundle.prototype.path = function() { CPLog.warn("CFBundle.prototype.path is deprecated, use CFBundle.prototype.bundlePath instead."); return this.bundlePath.apply(this, arguments); }; CFBundle.prototype.pathForResource = function(aResource, aType, aSubDirectory, localizationName) { return this.resourceURL(aResource, aType, aSubDirectory, localizationName).absoluteString(); }; GLOBAL(CFBundleCopyLocalizedString) = function (/*Bundle*/ bundle, key, value, tableName) { return CFCopyLocalizedStringWithDefaultValue(key, tableName, bundle, value, ""); } GLOBAL(CFBundleCopyBundleLocalizations) = function (/*Bundle*/ aBundle) { return [this._loadedLanguage]; } GLOBAL(CFCopyLocalizedString) = function (key, comment) { return CFCopyLocalizedStringFromTable(key, "Localizable", comment); } GLOBAL(CFCopyLocalizedStringFromTable) = function (key, tableName, comment) { return CFCopyLocalizedStringFromTableInBundle(key, tableName, CFBundleGetMainBundle(), comment); } GLOBAL(CFCopyLocalizedStringFromTableInBundle) = function (key, tableName, bundle, comment) { return CFCopyLocalizedStringWithDefaultValue(key, tableName, bundle, null, comment); } GLOBAL(CFCopyLocalizedStringWithDefaultValue) = function (key, tableName, bundle, value, comment) { var string; if (!tableName) tableName = "Localizable"; tableName += ".strings"; var localizableString = bundle._localizableStrings[tableName]; string = localizableString ? localizableString[key + comment] : null; return string || (value || key); } GLOBAL(CFBundleGetMainBundle) = function () { return CFBundle.mainBundle(); };