/*global XMLHttpRequest, setTimeout, document, navigator, ActiveXObject*/

"use strict";

(function () {
	var
		  undef_type = "undefined"
		, string_type = "string"
		, String_ctr = String
		, has_own_prop = Object.prototype.hasOwnProperty
		, load_queues = {}
		, localizations = {}
		, FALSE = !1
		// the official format is application/vnd.oftn.l10n+json, though l10n.js will also
		// accept application/x-l10n+json and application/l10n+json
		, l10n_js_media_type = /^\s*application\/(?:vnd\.oftn\.|x-)?l10n\+json\s*(?:$|;)/i
		, XHR
		, $to_locale_string = "toLocaleString"
		, $to_lowercase = "toLowerCase"
	
	, array_index_of = Array.prototype.indexOf || function (item) {
		var
			  len = this.length
			, i   = 0
		;
		
		for (; i < len; i++) {
			if (i in this && this[i] === item) {
				return i;
			}
		}
		
		return -1;
	}
	, request_JSON = function (uri) {
		var req = new XHR();
		
		// sadly, this has to be blocking to allow for a graceful degrading API
		req.open("GET", uri, FALSE);
		req.send(null);
		
		if (req.status !== 200) {
			// warn about error without stopping execution
			setTimeout(function () {
				// Error messages are not localized as not to cause an infinite loop
				var l10n_err = new Error("Unable to load localization data: " + uri + " Status error: " + req.status);
				l10n_err.name = "Localization Error";
				throw l10n_err;
			}, 0);
			
			return {};
		} else {
			return JSON.parse(req.responseText);
		}
	}
	, load = String_ctr[$to_locale_string] = function (data) {
		// don't handle function[$to_locale_string](indentationAmount:Number)
		if (arguments.length > 0 && typeof data !== "number") {
			if (typeof data === string_type) {
				load(request_JSON(data));
			} else if (data === FALSE) {
				// reset all localizations
				localizations = {};
			} else {
				// Extend current localizations instead of completely overwriting them
				for (var locale in data) {
					if (has_own_prop.call(data, locale)) {
						var localization = data[locale];
						locale = locale[$to_lowercase]();
						
						if (!(locale in localizations) || localization === FALSE) {
							// reset locale if not existing or reset flag is specified
							localizations[locale] = {};
						}
						
						if (localization === FALSE) {
							continue;
						}
						
						// URL specified
						if (typeof localization === string_type) {
							if (String_ctr.locale[$to_lowercase]().indexOf(locale) === 0) {
								localization = request_JSON(localization);
							} else {
								// queue loading locale if not needed
								if (!(locale in load_queues)) {
									load_queues[locale] = [];
								}
								load_queues[locale].push(localization);
								continue;
							}
						}
						
						for (var message in localization) {
							if (has_own_prop.call(localization, message)) {
								localizations[locale][message] = localization[message];
							}
						}
					}
				}
			}
		}
		// Return what function[$to_locale_string]() normally returns
		return Function.prototype[$to_locale_string].apply(String_ctr, arguments);
	}
	, process_load_queue = function (locale) {
		var queue = load_queues[locale],
		i = 0,
		len = queue.length;
		
		for (; i < len; i++) {
			var localization = {};
			localization[locale] = request_JSON(queue[i]);
			load(localization);
		}
		
		delete load_queues[locale];
	}

	;
	
	if (typeof XMLHttpRequest === undef_type && typeof ActiveXObject !== undef_type) {
		var AXO = ActiveXObject;
		
		XHR = function () {
			try {
				return new AXO("Msxml2.XMLHTTP.6.0");
			} catch (xhrEx1) {}
			try {
				return new AXO("Msxml2.XMLHTTP.3.0");
			} catch (xhrEx2) {}
			try {
				return new AXO("Msxml2.XMLHTTP");
			} catch (xhrEx3) {}
		
			throw new Error("XMLHttpRequest not supported by this browser.");
		};
	} else {
		XHR = XMLHttpRequest;
	}
	
	if (!String_ctr.locale) {
		if (typeof navigator !== undef_type) {
			var nav = navigator;
			String_ctr.locale = nav.language || nav.userLanguage || "";
		} else {
			String_ctr.locale = "";
		}
	}
	
	if (typeof document !== undef_type) {
		var
			  elts = document.getElementsByTagName("link")
			, i = elts.length
		;
		
		while (i--) {
			var
				  elt = elts[i]
				, rel = (elt.getAttribute("rel") || "")[$to_lowercase]().split(/\s+/)
			;
			
			if (l10n_js_media_type.test(elt.type)) {
				if (array_index_of.call(rel, "localizations") !== -1) {
					// multiple localizations
					load(elt.getAttribute("href"));
				} else if (array_index_of.call(rel, "localization") !== -1) {
					// single localization
					var localization = {};
					localization[(elt.getAttribute("hreflang") || "")[$to_lowercase]()] =
						elt.getAttribute("href");
					load(localization);
				}
			}
		}
	}
	
	String_ctr.prototype[$to_locale_string] = function () {
		var
			  parts = String_ctr.locale[$to_lowercase]().split("-")
			, i = parts.length
			, this_val = this.valueOf()
		;
		var isfound = false;
		// Iterate through locales starting at most-specific until localization is found
		do {
			var locale = parts.slice(0, i).join("-");
			// load locale if not loaded
			if (locale in load_queues) {
				process_load_queue(locale);
			}
			if (locale in localizations && this_val in localizations[locale]) {
				return localizations[locale][this_val];
			}
		}
		while (i--);
		
        var returnval;
        this_val in localizations["en"] ? returnval = localizations["en"][this_val] : returnval = this_val;
        return returnval;
		//return this_val;
	};
}());

