/****************************
 * Node Types http://www.w3schools.com/dom/dom_nodetype.asp
 * NodeType 	Named Constant
 * 1 	ELEMENT_NODE
 * 2 	ATTRIBUTE_NODE
 * 3 	TEXT_NODE
 * 4 	CDATA_SECTION_NODE
 * 5 	ENTITY_REFERENCE_NODE
 * 6 	ENTITY_NODE
 * 7 	PROCESSING_INSTRUCTION_NODE
 * 8 	COMMENT_NODE
 * 9 	DOCUMENT_NODE
 * 10 	DOCUMENT_TYPE_NODE
 * 11 	DOCUMENT_FRAGMENT_NODE
 * 12 	NOTATION_NODE
 ****************************/

var TongWen = (function () {
	var
		version  = "0.2",         // 版本

		flagSimp = "simplified",  // 簡體
		flagTrad = "traditional", // 繁體

		zhEncodesSimp = ["gb2312", "gbk", "x-gbk", "gb18030", "hz-gb-2312", "iso-2022-cn"],
		zhEncodesTrad = ["big5", "big5-hkscs", "x-euc-tw"],
		zhEncodesAll  = zhEncodesSimp.concat(zhEncodesTrad, ["utf-7", "utf-8", "utf-16le", "x-user-defined"]),

		t2s = {},                 // 繁轉簡 對照表
		s2t = {},                 // 簡轉繁 對照表

		maxSTLen = 1,             // 簡轉繁 最長的詞句
		maxTSLen = 1,             // 繁轉簡 最長的詞句

		curZhFlag      = "",      // 目前網頁編碼
		toolbarShow    = false;
		return init();
		// TongWen = window.TongWen = init();

	function init() {
		var that = this;
		toolbarId = "tongwen-" + Math.ceil(Math.random() * 1000000);

		if (document.addEventListener) {
			document.addEventListener("DOMContentLoaded", function () { winLoad() }, false);
		} else if (window.attachEvent) {
			window.attachEvent("onload", winLoad);
		}

		return {
			version    : version,
			flagSimp   : flagSimp,
			flagTrad   : flagTrad,
			addS2TTable: addS2TTable,
			addT2STable: addT2STable,
			convert    : convert,
			transPage  : transPage,
			trans2Trad : trans2Trad,
			trans2Simp : trans2Simp,
			transAuto  : transAuto
		}
	}

	function winLoad() {
		// createToolbar();
	}

	function createToolbar() {
		var node = document.createElement("div");
		node.style.width = "200px";
		node.style.height = "100px";
		node.style.position = "fixed";
		node.style.top = "10px";
		node.style.right = "10px";
		node.style.backgroundColor = "#F00";
		document.body.appendChild(node);
	}

// =============================================================================

	// 新增 簡轉繁 對照表
	function addS2TTable(table) {
		for (var i in table) {
			maxSTLen = Math.max(maxSTLen, table[i].length);
			s2t[i] = table[i];
		}
	}

	// 新增 繁轉簡 對照表
	function addT2STable(table) {
		for (var i in table) {
			maxTSLen = Math.max(maxTSLen, table[i].length);
			t2s[i] = table[i];
		}
	}

	function setZhFlag(doc, zhflag) {
		doc.documentElement.setAttribute("zhtongwen", zhflag);
	};

	function getZhFlag(doc) {
		var zhflag = "";
		if (doc && doc.documentElement) {
			zhflag = doc.documentElement.getAttribute("zhtongwen");
			if (zhflag == null) zhflag = "";
		}
		return zhflag;
	};

	// 繁簡轉換
	function convert(str, zhflag) {
		var leng = 4;
		var zmap = null;

		if (zhflag == flagSimp) {
			// 繁轉簡
			zmap = t2s;
			leng = Math.min(maxTSLen, str.length);
		} else {
			// 簡轉繁
			zmap = s2t;
			leng = Math.min(maxSTLen, str.length);
		}

		// 單字轉換
		str = str.split("");
		for (var i = 0, c = str.length; i < c; i++) {
			str[i] = zmap[str[i]] || str[i];
		}
		str = str.join("");

		// 詞彙轉換
		var txt = "", s = "", bol = true;
		for (var i = 0, c = str.length; i < c;) {
			bol = true;
			for (var j = leng; j > 1; j--) {
				s = str.substr(i, j);
				if (s in zmap) {
					txt += zmap[s];
					i += j;
					bol = false;
					break;
				}
			}

			if (bol) {
				txt += str.substr(i, 1);
				i++;
			}
		}
		if (txt != "") str = txt;
		return str;
	}

	function parseTree(doc, zhflag) {
		var nodes = [];
		var frame = [];

		function step() {
			if (nodes.length <= 0) {
				if (frame.length <= 0) {
					return;
				} else {
					transPage(frame.shift(), zhflag);
				}
			}
			var node = nodes.shift();
			var attr = null;

			// Node Types http://www.w3schools.com/dom/dom_nodetype.asp
			switch (node.nodeType) {
				case 1: // ELEMENT_NODE
					switch (node.nodeName.toLowerCase()) {
						case "frame" :
						case "iframe":
							if (typeof node.contentDocument != "undefined") {
								transPage(node.contentDocument, zhflag);
								// frame.push(node.contentDocument);
							} else if ((typeof node.contentWindow != "undefined") && (typeof node.contentWindow.document != "undefined")) {
								transPage(node.contentWindow.document, zhflag);
								// frame.push(node.contentWindow.document);
							}
							// transPage(node.contentDocument || node.contentWindow.document, zhflag);
							// frame.push(node.contentDocument || node.contentWindow.document);
							break;
						case "embed"   :
						case "object"  :
						case "script"  :
						case "style"   :
						case "title"   :
						case "br"      :
						case "hr"      :
						case "noscript":
						case "script"  :
							break;
						case "img":
							attr = node.getAttribute("title");
							if (attr != null) {
								node.setAttribute("title", convert(attr, zhflag));
							}
							attr = node.getAttribute("alt");
							if (attr != null) {
								node.setAttribute("alt", convert(attr, zhflag));
							}
							break;
						case "input":
							switch(node.type.toLowerCase()){
								case "button":
								case "submit":
								case "reset":
								// case "text": // keep org value
									if (node.value.length > 0) {
										node.value = convert(node.value, zhflag);
									}
									break;
								default:
							}
							break;
						case "option":
							if (node.text.length > 0) {
								node.text = convert(node.text, zhflag);
							}
							break;
						default:
							attr = node.getAttribute("title");
							if (attr != null) {
								node.setAttribute("title", convert(attr, zhflag));
							}
							if (node.hasChildNodes()) {
								for (var i = 0, c = node.childNodes.length; i < c; i++) {
									if ("139".indexOf(node.childNodes[i].nodeType) < 0) continue;
									nodes.push(node.childNodes[i]);
								}
							}
							break;
					}
					break;
				case 3: // TEXT_NODE
					var val = node.nodeValue;
					if (val.length > 0) {
						node.nodeValue = convert(val, zhflag);
					}
					step();
					break;
				case 9: // DOCUMENT_NODE
					if (node.hasChildNodes()) {
						for (var i = 0, c = node.childNodes.length; i < c; i++) {
							if ("139".indexOf(node.childNodes[i].nodeType) < 0) continue;
							nodes.push(node.childNodes[i]);
						}
					}
					break;
			}

			if ((nodes.length > 0) || (frame.length > 0)) {
				setTimeout(step, 1);
			}
		}

		nodes.push(doc);
		setTimeout(step, 1);
	}

	function transPage(doc, zhflag) {
		curZhFlag = zhflag;
		setZhFlag(doc, zhflag);
		doc.title = convert(doc.title, zhflag);
		parseTree(doc, zhflag);
	}

	function trans2Trad(doc) {
		transPage(doc || document, flagTrad);
	}

	function trans2Simp(doc) {
		transPage(doc || document, flagSimp);
	}

	function transAuto(doc) {
		var curDoc  = doc || document;
		var charset = curDoc.characterSet.toLowerCase();

		curZhFlag = getZhFlag(doc);
		if (curZhFlag == "") {
			var lang = curDoc.documentElement.getAttribute("lang");
			if (lang != null) {
				switch (lang.toLowerCase()) {
					case "zh-tw":
					case "zh-hk":
						curZhFlag = flagSimp;
						break;
					case "zh-cn":
						curZhFlag = flagTrad;
						break;
				}
			}

			if ((curZhFlag == "") && (zhEncodesAll.indexOf(charset) >= 0)) {
				curZhFlag = (zhEncodesTrad.indexOf(charset) >= 0) ? flagSimp : flagTrad;
			}
		} else {
			curZhFlag = (curZhFlag == flagTrad) ? flagSimp : flagTrad;
		}
		if (curZhFlag == "") return;

		transPage(curDoc, curZhFlag);
	}

})();

