/**
 * @name CeL function for source code.
 * @fileoverview 本檔案包含了處理 source code/text 的 functions。
 * @since
 */
// More examples: see /_test suite/test.js
'use strict';
// --------------------------------------------------------------------------------------------
// 不採用 if 陳述式,可以避免 Eclipse JSDoc 與 format 多縮排一層。
typeof CeL === 'function' && CeL.run({
	// module name
	name : 'data.code',
	// .set_bind()
	require : 'data.native.',
	// 設定不匯出的子函式。
	// no_extend : '*',
	// 為了方便格式化程式碼,因此將 module 函式主體另外抽出。
	code : module_code
});
function module_code(library_namespace) {
	// nothing required
	/**
	 * null module constructor
	 * 
	 * @class 處理 source code 的 functions
	 */
	var _// JSDT:_module_
	= function() {
		// null module constructor
	};
	/**
	 * for JSDT: 有 prototype 才會將之當作 Class
	 */
	_// JSDT:_module_
	.prototype = {};
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	// CamelCase to embedded_underscore/Snake case (underscore-based style) or
	// even hyphenated name
	function Camel_to_underscore(identifier, separator) {
		if (!separator)
			separator = '_';
		return identifier.replace(new RegExp('[\\' + separator + ']', 'g'),
				separator + separator).replace(/[A-Z]/g, function($0) {
			return separator + $0.toLowerCase();
		});
	}
	_.to_underscore = Camel_to_underscore;
	// underscore-based style to CamelCase
	function underscore_to_CamelCase(identifier, separator) {
		if (!separator)
			separator = '_';
		return identifier.replace(
				new RegExp('\\' + separator + '([a-zA-Z])', 'g'),
				function($0, $1) {
					return $1.toUpperCase();
				}).replace(new RegExp('[\\' + separator + ']{2}', 'g'),
				separator);
	}
	library_namespace.set_method(String.prototype, {
		to_underscore : library_namespace.set_bind(Camel_to_underscore),
		to_Camel : library_namespace.set_bind(underscore_to_CamelCase)
	});
	// @see to_hyphenated() @ interact.DOM
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	/**
	 * 類似 sprintf,處理 escape sequence 字串之 function。
	 * 
	 * TODO: http://numeraljs.com/
	 * 
	 * @example 
	 * 
	 * 
	 * 
	 * @param {String}string
	 *            欲格式化之字串 / source text.
	 * @param {Object|String|Function}[options]
	 *            附加參數/設定選擇性/特殊功能與選項: {
	 *            {character}escape: escape character,
	 *            {Object}escape_length: escape sequence length,
	 *            {Function}handler: 處理 source text (非 escape sequence) 之
	 *            function,
	 *            {Function}escape_handler: 處理 escape sequence 之 function.
 }
	 * 
	 * @returns {Array} source text list:
	 *          [source text, escape sequence, source text, escape sequence, ..]
	 */
	function parse_escape(string, options) {
		var
		/**
		 * 搜索到匹配之部分。
		 */
		matched,
		/**
		 * 搜索之 pattern。
		 * 
		 * @type {RegExp}
		 */
		parse_RegExp,
		/**
		 * 下次檢索的起始點。
		 * 
		 * @type {Integer}
		 */
		last_index = 0,
		/**
		 * escape_character
		 * 
		 * @see escape
		 *      character
		 * 
		 * @type {character}
		 */
		e_c = '\\',
		/**
		 * escape sequence length.
		 * default: 1.
		 * 為處理不定長 escape sequence. 這裡僅列出需要特別注意的。
		 * 
		 * @type {Object}
		 */
		e_l = {
			// TODO: [\d],
			u : 4,
			U : 8,
			x : 2
		},
		/**
		 * handle function.
		 * 處理 source text (非 escape sequence) 之 function。
		 * 
		 * @type {Function}
		 */
		handler = undefined,
		/**
		 * Single Character Escape Sequences
		 * 
		 * !see https://en.wikipedia.org/wiki/Escape_sequences_in_C
		 * 
		 * @type {Object}
		 */
		escape_sequences = {
			u : to_char,
			U : to_char,
			x : to_char,
			// '"' : '\"', "'" : "\'", '\\' : '\\',
			b : '\b',
			t : '\t',
			n : '\n',
			v : '\v',
			f : '\f',
			r : '\r'
		},
		/**
		 * escape sequence handle function.
		 * 處理 escape sequence 之 function.
		 * 
		 * @type {Function}
		 */
		e_s_handler = function(s, a) {
			library_namespace.debug(s + ': additional [' + a + '], ', 6);
			if (s in escape_sequences) {
				var f = escape_sequences[s];
				s = typeof f === 'function' ? f(s, a) : f;
			}
			return s;
		},
		/**
		 * 回傳之 source text list:
		 * [source text, escape sequence, source text, escape sequence, ..]
		 * 
		 * @type {Array}
		 */
		source_text_list = [];
		/**
		 * Unicode to character.
		 * 
		 * @param {character}c
		 *            escape sequence 的種類: x, u, U, ..
		 * @param {String}x
		 *            hexadecimal digits /[\da-f]/i
		 * 
		 * @returns {character} character
		 */
		function to_char(c, x) {
			library_namespace.debug('U+' + x + ': ['
					+ String.fromCharCode(parseInt(x, 16)) + ']', 6);
			return String.fromCharCode(parseInt(x, 16));
		}
		/**
		 * 處理匹配之部分:
		 * [source text, escape sequence]
		 * 
		 * @param {String}s
		 *            source text
		 * @param {String}e_s
		 *            escape sequence
		 */
		function handle_slice(s, e_s) {
			library_namespace.debug(last_index + ': [' + s + ']|'
					+ (e_s || ''), 6);
			if (s && handler)
				s = handler(s);
			if (e_s) {
				var l, e = '';
				if (e_s in e_l) {
					e = string.substr(last_index, l = e_l[e_s]);
					library_namespace.debug('(' + l + ') [' + e_s + e + ']', 6);
					parse_RegExp.lastIndex = (last_index += l);
				}
				if (e_s_handler)
					e_s = e_s_handler(e_s, e);
				else if (e !== '')
					e_s += e;
				source_text_list.push(s, e_s);
			} else if (s)
				source_text_list.push(s);
		}
		// 前置處理。
		if (typeof options === 'string')
			e_c = options;
		else if (typeof options === 'function')
			handler = options;
		else if (library_namespace.is_Object(options)) {
			if (typeof options.escape === 'string')
				e_c = options.escape;
			if (typeof options.escape_length === 'object')
				e_l = options.escape_length;
			if (typeof options.handler === 'function')
				handler = options.handler;
			if (typeof options.escape_handler === 'function')
				e_s_handler = options.escape_handler;
		}
		if (e_c.length !== 1)
			throw new Error('The escape character [' + e_c
					+ '] is not single character!');
		parse_RegExp = new RegExp('([\\s\\S]*?)\\' + e_c + '(.)', 'g');
		library_namespace.debug('[' + string + ']', 6);
		while (matched = parse_RegExp.exec(string)) {
			last_index = parse_RegExp.lastIndex;
			handle_slice(matched[1], matched[2]);
		}
		// 處理剩下未匹配之部分。
		handle_slice(string.slice(last_index));
		return handler ? source_text_list.join('') : source_text_list;
	}
	_// JSDT:_module_
	.parse_escape = parse_escape;
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	if (false) {
		'2A1B' === CeL.extract_literals('${a}A${b}B', {
			a : 2,
			b : 1
		});
	}
	/**
	 * 模仿樣板字面值(Template literals)
	 * 
	 * TODO: extract_literals('${ ({$:1})["$"] }')
	 * 
	 * @param {String}template_string
	 *            樣板字串(template strings)
	 * @param {Object}key_value_pairs
	 *            變數值 {key:value}
	 * @param {Object}[options]
	 *            附加參數/設定選擇性/特殊功能與選項
	 * 
	 * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals
	 */
	function extract_literals(template_string, key_value_pairs, options) {
		if (library_namespace.gettext)
			template_string = library_namespace.gettext(template_string);
		template_string = template_string.replace_till_stable(/\${([^{}]+?)}/,
		//
		function(all, expression) {
			expression = expression.trim();
			if (expression in key_value_pairs) {
				return key_value_pairs[expression];
			}
			return all;
		});
		return template_string;
	}
	_// JSDT:_module_
	.extract_literals = extract_literals;
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	// TODO
	// format (escape sequence / conversion specifications) parser
	// functional keyword,
	function format_parser(escape_character, command_set, options) {
		if (typeof escape_character !== 'string' || !escape_character
				|| !library_namespace.is_Object(command_set))
			return;
		function get_pattern() {
			var pattern = [];
			for ( var i in command_set) {
				pattern.push(command_set[i] && command_set[i].pattern || i);
			}
			return pattern.join('|');
		}
		var search;
		if (options) {
			if (!library_namespace.is_Object(this.options = options))
				options = {
					search : options
				};
			search = options.search;
		} else
			this.options = false;
		if (!search) {
			// RegExp punctuators, reserved words
			escape_character = escape_character.replace(
					/([!?:.*+-^${}()\\\-\[\]])/g, '\\$1');
			search = new RegExp('((?:[^' + escape_character + ']+|'
					+ escape_character + '{2})+)(' + escape_character + ')('
					+ get_pattern() + ')', 'g');
		} else if (!library_namespace.is_type(search, 'RegExp')) {
			search = new RegExp(('' + search)
					.replace(/pattern/g, get_pattern()), 'g');
		}
		this.search = search;
		return format_parser.default_parser.bind(this);
	}
	// parser|parser array
	format_parser.default_parser = function(object, format, usage) {
		var search = this.search, command_set = this.command_set, options = this.options,
		// 處理整段 matched 的函數。
		parse_matched = options.parser,
		// 處理一般字串的函數。
		normal_parser = options.normal_parser,
		// main-loop 所需。
		matched, last_index = 0, command, result = [];
		// 為了 g 初始化. 或者設定 .lastIndex = 0 ?
		search.exec('');
		while (matched = search.exec(format)) {
			last_index = search.lastIndex;
			if (parse_matched)
				result.push(parse_matched.call(this, object, matched, usage));
			else {
				// matched = [matched slice (normal + escape sequence), normal,
				// escape character, format pattern, format command];
				// 處理一般字串。
				result.push(normal_parser ? normal_parser(object, matched[1],
						usage) : matched[1]);
				// 處理一般 format。
				command = matched[4] || matched[3];
				result.push(command in command_set ? command_set[command].call(
						object, matched[3]) : matched[2] + matched[3]);
			}
		}
		// 加入最後一段。
		matched = format.slice(last_index);
		result.push(normal_parser ? normal_parser(object, matched, usage)
				: matched);
		return result.join('');
	};
	format_parser.default_parser.constructor = format_parser;
	format_parser.prototype.concat = function(parser) {
		// TODO
		throw new Error(1,
				'format_parser.prototype.concat: Not Yet Implemented!');
	};
	format_parser.prototype.extend = function() {
		var new_parser = new format_parser(this);
	};
	function hex_to_Unicode() {
		// TODO
		throw new Error(1, 'hex_to_Unicode: Not Yet Implemented!');
	}
	if (false) {
		// backslash escape sequence parser
		var backslash_parser = new format_parser('\\', {
			u : {
				pattern : /[\da-z]{4}/i,
				handler : hex_to_Unicode
			},
			U : {
				pattern : /[\da-z]{8}/i,
				handler : hex_to_Unicode
			},
			x : {
				pattern : /[\da-z]{2}/i,
				handler : hex_to_Unicode
			},
			// '"' : '\"', "'" : "\'", '\\' : '\\',
			b : '\b',
			t : '\t',
			n : '\n',
			v : '\v',
			f : '\f',
			r : '\r'
		});
		// sprintf-like format parser. % conversion specifications
		var sprintf = new format_parser('%', {
			// 數字
			d : function() {
				return parseInt(this.valueOf());
			},
			s : function() {
				return String(this.valueOf());
			}
		}, {
			// replace '[.]'
			search : /%([+\-]?)(\d{0,3})(?:\.(\d{1,2}))([.])/,
			// pre-parser
			normal_parser : backslash_parser
		});
		var extend_sprintf = sprintf.extend('%', {
			// 數字
			z : function() {
			}
		}, {
			// replace '[.]'
			search : /%([+\-]?)(\d{0,3})(?:\.(\d{1,2}))([.])/
		});
	}
	function set_toString(Class, format_parser, special_condition) {
		if (!Class || typeof format_parser !== 'function')
			return;
		// 以指定 format 轉換 Class 之內容成 string。
		var old_toString = Class.prototype.toString;
		// format 用途:i18n|不同領域、不同產業採用不同 format
		Class.prototype.toString = function(format, usage) {
			if (!argument.length)
				return old_toString.call(this);
			if (typeof format === 'number' && special_condition)
				format = typeof special_condition === 'object' ? special_condition[format]
						: typeof special_condition === 'function' ? special_condition(format)
								: format;
			return format_parser.call(this, format, usage);
		};
		return old_toString;
	}
	if (false) {
		set_toString(Date, backslash_parser.extend('%', {
			// 完整年份(四位數的數字,如2000)
			Y : function() {
				return this.getFullYear();
			},
			// 月份 (1-12)。
			m : function() {
				return 1 + this.getMonth();
			}
		}, {
			search : /%([+\-]?)(\d{0,3})(?:\.(\d{1,2}))([.])/
		}));
		set_toString(Number);
		set_toString(library_namespace.quotient, backslash_parser.extend('%', {
			// numerator
			n : function() {
				return this.n;
			},
			// denominator
			d : function() {
				return this.d;
			}
		}, {
			search : /%([+\-]?)(\d{0,3})(?:\.(\d{1,2}))([.])/
		}));
	}
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	var is_controller = library_namespace.is_Object;
	// 處理非巢式嵌套格式處理器。
	// 2013/1/27 20:9:34
	function unnested_formatter(parameters, start, end) {
		var formatter = {
			start : typeof start === 'function' ? start : start ? function() {
				return start;
			} : function(values) {
				return values.join('');
			},
			parameters : parameters
		};
		if (typeof end === 'function')
			formatter.end = end;
		else if (end)
			formatter.end = function() {
				return end;
			};
		return unnested_formatter_convert.bind(formatter);
	}
	// 將改變 status,不改變 controller, parameters.
	function unnested_formatter_change_status(status, controller, _this,
			controller_is_status) {
		var type, value, parameter, parameter_result, changed_value = typeof _this.end === 'function' ? []
				: false, changed_to_value = [];
		for (type in controller)
			if ((value = controller[type]) !== undefined
					&& (parameter = _this.parameters[type])) {
				parameter_result = controller_is_status ? value
				//
				: typeof parameter === 'function' ? parameter(value, type)
						: parameter + value;
				if (parameter_result != status[type]) {
					// 僅處理有改變的值。
					changed_value && changed_value.push(type);
					changed_to_value.push(status[type] = parameter_result);
				}
			}
		// start
		parameter_result = changed_to_value.length ? _this
				.start(changed_to_value) : '';
		return changed_value && changed_value.length ?
		// close + start
		String(_this.end(changed_value)) + parameter_result
		// start only.
		: parameter_result;
	}
	function unnested_formatter_convert_Array(format_Array, _this, meta_status,
			item_processor) {
		var index = 0, length = format_Array.length, formatted_result = [], item,
		//
		status_now = Object.create(null),
		// 囤積的 controller。
		controller, cloned;
		if (meta_status)
			// duplicate meta_status.
			Object.assign(status_now, meta_status);
		for (; index < length; index++) {
			item = format_Array[index];
			// 先讓 item_processor 處理一下。
			if (item_processor && typeof item !== 'object')
				item = item_processor(item);
			if (is_controller(item)) {
				if (controller) {
					if (!cloned) {
						controller = cloned = Object.assign(
								Object.create(null), controller);
						// cloned = true;
					}
					Object.assign(controller, item);
				} else
					// 不直接 clone,減少 copy 次數。
					controller = item;
			} else {
				// 連續的 controller 只有在必要時(最後一個),才處理。
				if (controller) {
					formatted_result.push(unnested_formatter_change_status(
							status_now, controller, _this));
					controller = cloned = false;
				}
				// need test if item === null, undefined?
				formatted_result
						.push(Array.isArray(item) ? unnested_formatter_convert_Array(
								item, _this, status_now)
								: item);
			}
		}
		// 回復狀態至進入前。
		if (meta_status)
			formatted_result.push(unnested_formatter_change_status(status_now,
					meta_status, _this, true));
		return formatted_result.join('');
	}
	// front-end.
	function unnested_formatter_convert(format_structure, initial_controller,
			item_processor) {
		if (!Array.isArray(format_structure))
			return typeof format_structure === 'string' ? format_structure
					: format_structure || format_structure === 0 ? ''
							+ format_structure : '';
		return unnested_formatter_convert_Array(format_structure, this,
				initial_controller
						&& unnested_formatter_change_status(
								Object.create(null), initial_controller, this),
				typeof item_processor === 'function' && item_processor);
	}
	_.unnested_formatter = unnested_formatter;
	// ---------------------------------------------------------------------//
	/**
	 * create instence
	 * 
	 * @seeASS Tags - Aegisub Manual, ASS(Advanced SubStation Alpha)@wiki - トップページ
	 */
	var ass_tag = new unnested_formatter({
		italics : function(value) {
			return '\\i' + (value ? 1 : 0);
		},
		bold : function(value) {
			return '\\b' + (isNaN(value) ? value ? 1 : 0 : value);
		},
		underlined : function(value) {
			return '\\u' + (value ? 1 : 0);
		},
		// striked out
		striked : function(value) {
			return '\\s' + (value ? 1 : 0);
		},
		// Border size
		border : '\\bord',
		border_x : '\\xbord',
		border_y : '\\ybord',
		// TODO: shad,xshad,yshad,Blur edges,Letter spacing,Text rotation,Text
		// shearing,alpha,Karaoke effect,Wrap style,position,Movement,Rotation
		// origin,Fade,Animated transform,Clip,Drawing tags
		font_name : '\\fn',
		font_size : '\\fs',
		font_scale_x : '\\fscx',
		font_scale_y : '\\fscy',
		font_encoding : '\\fe',
		color : function(value) {
			return '\\1c&H' + value + '&';
		},
		border_color : function(value) {
			return '\\3c&H' + value + '&';
		},
		shadow_color : function(value) {
			return '\\4c&H' + value + '&';
		},
		// Line alignment:
		// 789
		// 456
		// 123
		align : '\\an'
	}, function(array) {
		return '{' + array.join('') + '}';
	});
	ass_tag.reduce = function(tag) {
		return tag && typeof tag === 'string' ? tag.replace(/}{\\/g, '\\')
				.replace(/{[^{}]+\\r([}\\])/g, '{\\r$1').replace(
						/(\n[^{}\n]*){\\r}/g, '$1').replace(
						/\s*{\\r}(\s*\r?\n)/g, '$1').replace(
						/(\\r}[^{]*){\\r}/g, '$1').replace(
						/(\\r}[^{]*){\\r\\/g, '$1{') : tag;
	};
	_.ass_tag = ass_tag;
	// test: "{\1c&HEEFFEE&}colored text"
	// var tag = CeL.ass_tag([ { color : 'EEFFEE' }, 'colored text' ]);
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	// https://en.wikipedia.org/wiki/Tree_traversal
	// traversal algorithm
	// @see traversal_DOM_backward() @ CeL.interact.DOM
	function traversal(start_node, action, options) {
		// no for_node action: just get {Array}list
		action = function(node) {
			return exit;
		};
		options = {
			// https://en.wikipedia.org/wiki/Depth-first_search
			// pre-order, in-order and post-order depth-first traversal
			// https://en.wikipedia.org/wiki/Breadth-first_search
			type : 'breadth',
			// direction: forward, backward
			backward : true,
			start_node_is_root : true,
			terminate_node : node,
			// get next node
			next_node : function(node_now, index, parent) {
				return node;
			},
			filter : function(node) {
				return true;
			},
			// final action
			last : function() {
			}
		};
	}
	// ----------------------------------------------------------------------------------------------------------------------------------------------------------//
	return (_// JSDT:_module_
	);
}