Ver Fonte

211030 FAQ全功能

yukyo0821 há 3 anos atrás
pai
commit
0bd9e03f11

+ 51 - 7
backstage/blogs/routes.py

@@ -92,10 +92,10 @@ def refreshFur(itype):
 
 
 def newFur(irequest):
-    print(irequest.url)
+    # print(irequest.url)
     myType = {}
     myPath = ""
-    print(str(irequest.url).find("system_furniture"))
+    # print(str(irequest.url).find("system_furniture"))
     if str(irequest.url).find("system_furniture") >= 0:
         myType = SfType
         myPath = sfurniturePath
@@ -185,6 +185,7 @@ def edit_system_furniture():
                            furnitures=sortedData, length=len(furnitures),
                            )
 
+
 @blogs_app.route('/backstage/edit_news', methods=['GET'])
 def edit_news():
     response = requests.get('{}contents?url=/news'.format(PORTAL_SERVER))
@@ -192,10 +193,11 @@ def edit_news():
         sortedData = sorted(response.json(), key=lambda x: x['date'], reverse=True)
         #sortedData = sorted(sortedData, key=lambda x: x['type'])
         return render_template('news.html',
-                            title='消息與報導',
-                            legend='消息與報導',
-                            news=sortedData, length=len(response.json()),form=BlogCreateForm()
-                            )
+                               title='消息與報導',
+                               legend='消息與報導',
+                               news=sortedData, length=len(response.json()), form=BlogCreateForm()
+                               )
+
 
 @blogs_app.route('/backstage/edit_contact_us', methods=['GET'])
 def edit_contact_us():
@@ -204,9 +206,47 @@ def edit_contact_us():
         sortedData = sorted(response.json(), key=lambda x: x['date'], reverse=True)
         #sortedData = sorted(sortedData, key=lambda x: x['type'])
         return render_template('contact_us.html',
-                            title='聯絡我們',
+                               title='聯絡我們',
+                               )
+
+@blogs_app.route('/backstage/edit_faq', methods=['GET'])
+def edit_faq():
+    return render_template('frequently_asked_questions.html',
+                            title='常見問題',
                             )
 
+@blogs_app.route('/backstage/edit_contact_us_getemail', methods=['GET'])
+def edit_contact_us_getemail():
+    txt = ""
+    newEmail = request.args.get('newemail', type=str)
+    footerpath = UPLOAD_PATH_MAP[0][0] + \
+        "../../../bhouseWeb/themes/hugo-lamp/layouts/partials/footer.html"
+    with open(footerpath, encoding="utf-8") as inf:
+        txt = inf.read()
+
+    result = txt[txt.find('"mailto:')+8:txt.find('"',txt.find('"mailto:')+1)]
+
+    #sortedData = sorted(sortedData, key=lambda x: x['type'])
+    return result
+
+@blogs_app.route('/backstage/edit_contact_us_editemail', methods=['GET'])
+def edit_contact_us_editemail():
+    txt = ""
+    newEmail = request.args.get('newemail', type=str)
+    footerpath = UPLOAD_PATH_MAP[0][0] + \
+        "../../../bhouseWeb/themes/hugo-lamp/layouts/partials/footer.html"
+    with open(footerpath, encoding="utf-8") as inf:
+        txt = inf.read()
+
+    result = re.sub(r'"mailto:(\S*)"', '"mailto:' + newEmail + '"', txt)
+
+    with open(footerpath, 'w' , encoding="utf-8") as inf:
+        inf.write(result)      
+
+    #sortedData = sorted(sortedData, key=lambda x: x['type'])
+    return "修改成功" + result
+
+
 @blogs_app.route('/backstage/blog/create/', methods=['POST'])
 def create():
     transcat = ""
@@ -275,6 +315,7 @@ categories: ["{}"]\n\
     print("11")
     return Response("你好", 200)
 
+
 @blogs_app.route('/backstage/news/create/', methods=['POST'])
 def createNews():
     form = BlogCreateForm()
@@ -300,16 +341,19 @@ image: ""\n\
 
     return create_content(data, form.image.data)
 
+
 @blogs_app.route('/backstage/blog/remove', methods=['POST'])
 def remove():
     remove_content()
     return redirect(url_for('blogs.blog_list'))
 
+
 @blogs_app.route('/backstage/news/remove', methods=['POST'])
 def removeNews():
     remove_content()
     return redirect(url_for('blogs.edit_news'))
 
+
 @blogs_app.route('/backstage/utils', methods=['GET'])
 def transService():
     # print(request.args["trantext"])

+ 1635 - 0
backstage/static/js/htmleditor.js

@@ -0,0 +1,1635 @@
+/*!
+ * http://suyati.github.io/line-control
+ * LineControl 1.1.0
+ * Copyright (C) 2014, Suyati Technologies
+ * This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version.
+This program 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 General Public License for more details.
+You should have received a copy of the GNU General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ *
+*/
+
+(function( $ ){
+	var editorObj;
+	var methods = {
+		saveSelection: function() {
+			//Function to save the text selection range from the editor
+			$(this).data('editor').focus();
+		    if (window.getSelection) {
+		        sel = window.getSelection();
+		        if (sel.getRangeAt && sel.rangeCount) {
+		            $(this).data('currentRange', sel.getRangeAt(0));
+		        }
+		    } else if (document.selection && document.selection.createRange) {
+		        $(this).data('currentRange',document.selection.createRange());
+		    }
+		    else
+		    	$(this).data('currentRange', null);
+		},
+
+		restoreSelection: function(text,mode) {
+			//Function to restore the text selection range from the editor
+			var node;
+			typeof text !== 'undefined' ? text : false;
+			typeof mode !== 'undefined' ? mode : "";
+			var range = $(this).data('currentRange');
+		    if (range) {
+		        if (window.getSelection) {
+		        	if(text){
+		            	range.deleteContents();
+		            	if(mode=="html")
+	            		{
+    			            var el = document.createElement("div");
+				            el.innerHTML = text;
+				            var frag = document.createDocumentFragment(), node, lastNode;
+				            while ( (node = el.firstChild) ) {
+				                lastNode = frag.appendChild(node);
+				            }
+				            range.insertNode(frag);
+	            		}
+		            	else
+            				range.insertNode( document.createTextNode(text) );
+
+		            }
+		            sel = window.getSelection();
+		            sel.removeAllRanges();
+		            sel.addRange(range);		            
+		        }
+		        else if (document.selection && range.select) {
+		            range.select();
+		            if(text)
+		            {
+		            	if(mode=="html")
+		            		range.pasteHTML(text);
+		            	else
+		            		range.text = text;
+		            }
+		        }
+		    }
+		},
+
+		restoreIESelection:function() {
+			//Function to restore the text selection range from the editor in IE
+			var range = $(this).data('currentRange');
+		    if (range) {
+		        if (window.getSelection) {
+		            sel = window.getSelection();
+		            sel.removeAllRanges();
+		            sel.addRange(range);
+		        } else if (document.selection && range.select) {
+		            range.select();
+		        }
+		    }
+		},
+
+		insertTextAtSelection:function(text,mode) {
+		    var sel, range, node ;
+		    typeof mode !== 'undefined' ? mode : "";
+		    if (window.getSelection) {
+		        sel = window.getSelection();
+		        if (sel.getRangeAt && sel.rangeCount) {
+		            range = sel.getRangeAt(0);
+		            range.deleteContents();
+		            var textNode = document.createTextNode(text); 
+		            
+		            if(mode=="html")
+		            { 
+		                var el = document.createElement("div");
+		                el.innerHTML = text;
+		                var frag = document.createDocumentFragment(), node, lastNode;
+		                while ( (node = el.firstChild) ) {
+		                    lastNode = frag.appendChild(node);
+		                }
+		                range.insertNode(frag);
+		            }
+		            else
+		            { 
+		            	range.insertNode(textNode);
+		            	range.selectNode(textNode);
+		            }
+		            sel.removeAllRanges();
+		            range = range.cloneRange();		            
+		            range.collapse(false);
+		            sel.addRange(range);
+		        }
+		    } else if (document.selection && document.selection.createRange) { 
+		        range = document.selection.createRange();
+		        range.pasteHTML(text);
+		        range.select();
+		    }
+		},
+
+		imageWidget: function(){
+			//Class for Widget Handling the upload of Files
+			var _idSuffix = this.attr("id");
+			var row = $('<div/>',{
+				"class":"row"
+			}).append($('<div/>',{
+				id :"imgErrMsg_" + _idSuffix
+			}));
+			var container = $('<div/>',{'class':"tabbable tabs-left"});
+			var navTabs = $('<ul/>',
+									{ class: "nav nav-tabs"
+							}).append($('<li/>',
+										{ class:"active"
+									}).append($('<a/>',{
+											"href":"#uploadImageBar_" + _idSuffix,
+											"data-toggle":"tab"
+										}).html("From Computer")
+							)).append($('<li/>').append($('<a/>',{
+											"href":"#imageFromLinkBar_" + _idSuffix,
+											"data-toggle":"tab"
+										}).html("From URL")));
+
+			var tabContent 		= $("<div/>", {class:"tab-content"});
+			var uploadImageBar  = $("<div/>",{
+				id: "uploadImageBar_" + _idSuffix,
+				class: "tab-pane active"
+			});
+
+			handleFileSelect = function(evt) {
+    			var files = evt.target.files; // FileList object
+				var output = [];
+				for (var i = 0, f; f = files[i]; i++) {
+					//Loop thorugh all the files
+					if(!f.type.match('image.*') || !f.name.match(/(?:gif|jpg|png|jpeg)$/)){ //Process only Images
+						methods.showMessage.apply(this,["imgErrMsg_" + _idSuffix,"Invalid file type"]);
+						continue;
+					}
+					var reader = new FileReader();
+					reader.onload = (function(imageFile){
+						return function(e){
+							//Render Thumnails
+							var li = $('<li/>',{class:"col-xs-12 col-sm-6 col-md-3 col-lg-3"});
+							var a = $('<a/>',{
+								href:"javascript:void(0)",
+								class:"thumbnail"
+							});
+							var image = $('<img/>',{
+								src:e.target.result,
+								title:escape(imageFile.name)
+							}).appendTo(a).click(function(){
+								$('#imageList_' + _idSuffix).data('current', $(this).attr('src'));
+								});
+							li.append(a).appendTo($('#imageList_' + _idSuffix));
+						}
+					})(f);
+					reader.readAsDataURL(f);					
+				}				
+			}
+			var chooseFromLocal = $('<input/>',{
+				type: "file",
+				class:"inline-form-control",
+				multiple: "multiple"
+			});
+			chooseFromLocal.on('change', handleFileSelect);
+			uploadImageBar.append(chooseFromLocal);
+			var imageFromLinkBar = $("<div/>",{
+				id: "imageFromLinkBar_" + _idSuffix,
+				class: "tab-pane"
+			});		
+			var getImageURL = $("<div/>", {class:"input-group"});
+			var imageURL = $('<input/>',{
+				type: "url",
+				class:'form-control',
+				id:"imageURL_" + _idSuffix,
+				placeholder: "Enter URL"
+			}).appendTo(getImageURL);
+			var getURL = $("<button/>",{
+				class:"btn btn-success",
+				type:"button"
+			}).html("Go!").click(function(){
+				var url = $('#imageURL_' + _idSuffix).val();
+				if(url ==''){
+					methods.showMessage.apply(this,["imgErrMsg_" + _idSuffix,"Please enter image url"]);
+					return false;
+				}
+				var li = $('<li/>',{class:"span6 col-xs-12 col-sm-6 col-md-3 col-lg-3"});
+				var a = $('<a/>',{
+					href:"javascript:void(0)",
+					class:"thumbnail"
+				});
+				var image = $('<img/>',{
+					src:url,
+				}).error(function(){
+				  	methods.showMessage.apply(this,["imgErrMsg_" + _idSuffix,"Invalid image url"]);
+				  	return false;
+				}).load( function() { $(this).appendTo(a).click(function(){
+					$('#imageList_' + _idSuffix).data('current', $(this).attr('src'));
+				});
+				li.append(a).appendTo($('#imageList_' + _idSuffix));
+			});
+			}).appendTo($("<span/>", {class:"input-group-btn form-control-button-right"}).appendTo(getImageURL));
+
+			imageFromLinkBar.append(getImageURL);
+			tabContent.append(uploadImageBar).append(imageFromLinkBar);
+			container.append(navTabs).append(tabContent);						
+
+			var imageListContainer = $("<div/>",{'class': 'col-xs-12 col-sm-12 col-md-12 col-lg-12'});
+			var imageList = $('<ul/>',{"class":"thumbnails padding-top list-unstyled",
+										"id": 'imageList_' + _idSuffix
+			}).appendTo(imageListContainer);
+			row.append(container).append(imageListContainer);
+			return row;
+		},
+
+		tableWidget: function(mode){
+			//Function to generate the table input form
+			var idExtn = "_" + $(this).attr("id");
+			if (typeof mode!=='undefined') {
+				idExtn = "_" + editorObj.attr("id") + "_Edt";
+			}
+			var tblCntr = $('<div/>',{ //Outer Container Div
+				class:"row-fluid"
+				}).append($('<div/>',{ //Err Message Div
+				 	id :"tblErrMsg"+idExtn 
+				})).append($('<form/>',{ //Form 
+					id:"tblForm"+idExtn 
+					}).append($('<div/>',{ //Inner Container Div
+						class:"row" 
+						}).append($('<div/>',{ //Left input Container Div
+							id :"tblInputsLeft"+idExtn,
+							class:"col-xs-12 col-sm-6 col-md-6 col-lg-6"
+							}).append($('<label/>',{ for:"tblRows"+idExtn,	text:"Rows"}
+							)).append($('<input/>',{
+								id:"tblRows"+idExtn,
+								type:"text",
+								class:"form-control form-control-width",
+								value:2
+							})).append($('<label/>',{ for:"tblColumns"+idExtn,	text:"Columns"}
+							)).append($('<input/>',{
+								id:"tblColumns"+idExtn,
+								type:"text",
+							 	class:"form-control form-control-width",
+							 	value:2
+							})).append($('<label/>',{ for:"tblWidth"+idExtn, text:"Width"}
+							)).append($('<input/>',{
+								id:"tblWidth"+idExtn,
+								type:"text",
+								class:"form-control form-control-width",
+								value:400
+							})).append($('<label/>',{ for:"tblHeight"+idExtn, text:"Height"}
+							)).append($('<input/>',{ 
+								id:"tblHeight"+idExtn,
+								type:"text",
+								class:"form-control form-control-width", 
+							}))
+						).append($('<div/>',{ //Right input Container Div
+							id :"tblInputsRight"+idExtn,
+							class:"col-xs-12 col-sm-6 col-md-6 col-lg-6"
+							}).append($('<label/>',{ for:"tblAlign"+idExtn, text:"Alignment"}
+							)).append($('<select/>',{ id:"tblAlign"+idExtn, class:"form-control form-control-width"}
+								).append($('<option/>',{ text:"Choose", value:""}
+								)).append($('<option/>',{ text:"Left", value:"left"}
+								)).append($('<option/>',{ text:"Center", value:"center"}
+								)).append($('<option/>',{ text:"Right",	value:"right"}))
+							).append($('<label/>',{	for:"tblBorder"+idExtn, text:"Border size"}
+							)).append($('<input/>',{ 
+								id:"tblBorder"+idExtn,
+								type:"text",
+								class:"form-control form-control-width",
+								value:1
+							})).append($('<label/>',{ for:"tblCellspacing"+idExtn,	text:"Cell spacing"}
+							)).append($('<input/>',{
+								id:"tblCellspacing"+idExtn,
+								type:"text", 
+								class:"form-control form-control-width",
+								value:1
+							})).append($('<label/>',{ for:"tblCellpadding"+idExtn,	text:"Cell padding"}
+							)).append($('<input/>',{
+								id:"tblCellpadding"+idExtn,
+								type:"text",
+								class:"form-control form-control-width",
+								value:1
+							}))
+						)
+					)
+				)																	
+			return tblCntr;
+		},
+
+		imageAttributeWidget: function(){
+
+			var edtTablecntr=$('<div/>',{ 
+				class:"row-fluid"}
+				).append($('<div/>',{ //Err Message Div
+				 	id :"imageErrMsg" 
+				})).append($('<input/>',{ 
+						id:"imgAlt",
+						type:"text",
+						class:"form-control form-control-link ",
+						placeholder:"Alt Text",
+					})).append($('<input/>',{
+						id:"imgTarget",
+						class:"form-control form-control-link ",
+						type:"text",
+						placeholder:"Link Target"
+					})).append($('<input/>',{
+						id:"imgHidden",
+						type:"hidden"						
+					}))
+					
+				return edtTablecntr;
+
+		},
+
+		getHTMLTable: function(tblRows,tblColumns,attributes){
+			//Function to generate html table. Supplied arguments: tablerows-no.of rows, no.of columns, table attributes.
+			var tableElement = $('<table/>',{ class:"table" });
+			for (var i = 0; i < attributes.length; i++){
+				if(attributes[i].value!=''){
+					if(attributes[i].attribute=="width" || attributes[i].attribute=="height")
+	                  	tableElement.css(attributes[i].attribute,attributes[i].value);
+					else
+						tableElement.attr(attributes[i].attribute,attributes[i].value);
+				}
+			}
+			for(var i=1; i<=tblRows; i++){
+				var tblRow = $('<tr/>');
+			 	for(var j=1; j<=tblColumns; j++){
+			 		var tblColumn = $('<td/>').html('&nbsp;');
+			 		tblColumn.appendTo(tblRow);
+			 	}				
+				tblRow.appendTo(tableElement);
+			}
+			return tableElement;
+		},
+
+		init : function( options )
+		{
+			if ($(this).attr("id") === undefined || $(this).attr("id") === "") {
+				$(this).attr("id", Date.now());
+			}
+			var fonts = { "Sans serif"	 : "arial,helvetica,sans-serif",
+						  "Serif"	 	 : "times new roman,serif",
+						  "Wide"	 	 : "arial black,sans-serif",
+						  "Narrow"	 	 : "arial narrow,sans-serif",
+						  "Comic Sans MS": "comic sans ms,sans-serif",
+						  "Courier New"  : "courier new,monospace",
+						  "Garamond"	 : "garamond,serif",
+						  "Georgia"	 	 : "georgia,serif",
+						  "Tahoma" 		 : "tahoma,sans-serif",
+						  "Trebuchet MS" : "trebuchet ms,sans-serif",
+						  "Verdana" 	 : "verdana,sans-serif"};
+
+			var styles = {  "Heading 1":"<h1>",
+							"Heading 2":"<h2>",
+							"Heading 3":"<h3>",
+							"Heading 4":"<h4>",
+							"Heading 5":"<h5>",
+							"Heading 6":"<h6>",
+							"Paragraph":"<p>" };
+
+			var fontsizes = {	"Small"	:"2",
+								"Normal":"3",
+								"Medium":"4",
+								"Large"	:"5",
+								"Huge"	:"6" };
+
+			var colors = [	{ name: 'Black', hex: '#000000' },
+							{ name: 'MediumBlack', hex: '#444444' },
+							{ name: 'LightBlack', hex: '#666666' },
+							{ name: 'DimBlack', hex: '#999999' },
+							{ name: 'Gray', hex: '#CCCCCC' },
+							{ name: 'DimGray', hex: '#EEEEEE' },
+							{ name: 'LightGray', hex: '#F3F3F3' },
+							{ name: 'White', hex: '#FFFFFF' },
+
+							{ name: 'libreak', hex: null },
+
+							{ name: 'Red', hex: '#FF0000' },
+							{ name: 'Orange', hex: '#FF9900' },
+							{ name: 'Yellow', hex: '#FFFF00' },
+							{ name: 'Lime', hex: '#00FF00' },
+							{ name: 'Cyan', hex: '#00FFFF' },
+							{ name: 'Blue', hex: '#0000FF' },
+							{ name: 'BlueViolet', hex: '#8A2BE2' },
+							{ name: 'Magenta', hex: '#FF00FF' },
+
+							{ name: 'libreak', hex: null },
+							
+							{ name: 'LightPink', hex: '#FFB6C1'},
+							{ name: 'Bisque', hex: '#FCE5CD'},
+							{ name: 'BlanchedAlmond', hex: '#FFF2CC'},
+							{ name: 'LightLime', hex: '#D9EAD3'},
+							{ name: 'LightCyan', hex: '#D0E0E3'},
+							{ name: 'AliceBlue', hex: '#CFE2F3'},
+							{ name: 'Lavender', hex: '#D9D2E9'},
+							{ name: 'Thistle', hex: '#EAD1DC'},
+
+							{ name: 'LightCoral', hex: '#EA9999' },
+							{ name: 'Wheat', hex: '#F9CB9C' },
+							{ name: 'NavajoWhite', hex: '#FFE599' },
+							{ name: 'DarkSeaGreen', hex: '#B6D7A8' },
+							{ name: 'LightBlue', hex: '#A2C4C9' },
+							{ name: 'SkyBlue', hex: '#9FC5E8' },
+							{ name: 'LightPurple', hex: '#B4A7D6' },
+							{ name: 'PaleVioletRed', hex: '#D5A6BD' },
+
+							{ name: 'IndianRed', hex: '#E06666' },
+							{ name: 'LightSandyBrown', hex: '#F6B26B' },
+							{ name: 'Khaki', hex: '#FFD966' },
+							{ name: 'YellowGreen', hex: '#93C47D' },
+							{ name: 'CadetBlue', hex: '#76A5AF' },
+							{ name: 'DeepSkyBlue', hex: '#6FA8DC' },
+							{ name: 'MediumPurple', hex: '#8E7CC3' },
+							{ name: 'MediumVioletRed', hex: '#C27BA0' },
+
+							{ name: 'Crimson', hex: '#CC0000' },
+							{ name: 'SandyBrown', hex: '#E69138' },
+							{ name: 'Gold', hex: '#F1C232' },
+							{ name: 'MediumSeaGreen', hex: '#6AA84F' },
+							{ name: 'Teal', hex: '#45818E' },
+							{ name: 'SteelBlue', hex: '#3D85C6' },
+							{ name: 'SlateBlue', hex: '#674EA7' },
+							{ name: 'VioletRed', hex: '#A64D79' },
+
+							{ name: 'Brown', hex: '#990000' },
+							{ name: 'Chocolate', hex: '#B45F06' },
+							{ name: 'GoldenRod', hex: '#BF9000' },
+							{ name: 'Green', hex: '#38761D' },
+							{ name: 'SlateGray', hex: '#134F5C' },
+							{ name: 'RoyalBlue', hex: '#0B5394' },
+							{ name: 'Indigo', hex: '#351C75' },
+							{ name: 'Maroon', hex: '#741B47' },
+
+							{ name: 'DarkRed', hex: '#660000' },
+							{ name: 'SaddleBrown', hex: '#783F04' },
+							{ name: 'DarkGoldenRod', hex: '#7F6000' },
+							{ name: 'DarkGreen', hex: '#274E13' },
+							{ name: 'DarkSlateGray', hex: '#0C343D' },
+							{ name: 'Navy', hex: '#073763' },
+							{ name: 'MidnightBlue', hex: '#20124D' },
+							{ name: 'DarkMaroon', hex: '#4C1130' } ];
+
+			var specialchars = [{ name:"Exclamation ", text:"!"},
+								{ name:"At", text:"@"},
+								{ name:"Hash", text:"#"},
+								{ name:"Percentage", text:"%"},
+								{ name:"Uppercase", text:"^"},
+								{ name:"Ampersand", text:"&"},
+								{ name:"Asterisk", text:"*"},
+								{ name:"OpenBracket", text:"("},
+								{ name:"CloseBracket", text:")"},
+								{ name:"Underscore", text:"_"},
+								{ name:"Hiphen", text:"-"},
+								{ name:"Plus", text:"+"},
+								{ name:"Equalto", text:"="},
+								{ name:"OpenSquareBracket", text:"["},
+								{ name:"CloseSquareBracket", text:"]"},
+								{ name:"OpenCurly", text:"{"},
+								{ name:"CloseCurly", text:"}"},
+								{ name:"Pipe", text:"|"},
+								{ name:"Colon", text:":"},
+								{ name:"Semicolon", text:";"},
+								{ name:"Single quote", text:"&#39;"},
+								{ name:"Double quote", text:"&#34;"},
+								{ name:"Left single curly quote", text:"&lsquo;"},
+								{ name:"right single curly quote", text:"&rsquo;"},
+								{ name:"Forward-slash", text:"&#47;"},
+								{ name:"Back-slash", text:"&#92;"},
+								{ name:"LessThan", text:"<"},
+								{ name:"GreaterThan", text:">"},
+								{ name:"QuestionMark", text:"?"},
+								{ name:"Tilda", text:"~"},
+								{ name:"Grave accent", text:"`"},
+								{ name:"Micron", text:"&micro;"},
+								{ name:"Paragraph sign", text:"&para;"},
+								{ name:"Plus/minus", text:"&plusmn;"},
+								{ name:"Trademark", text:"&trade;"},
+								{ name:"Copyright", text:"&copy;"},
+								{ name:"Registered", text:"&reg;"},
+								{ name:"Section", text:"&sect;"},
+								{ name:"right double angle quotes", text:"&#187;"},
+								{ name:"fraction one quarter", text:"&#188;"},
+								{ name:"fraction one half", text:"&#189;"},
+								{ name:"fraction three quarters", text:"&#190;"},
+								{ name:"Dollar", text:"$"},
+								{ name:"Euro", text:"&euro;"},
+								{ name:"Pound", text:"&pound;"},
+								{ name:"Yen", text:"&yen;"},
+								{ name:"Cent", text:"&#162;"},
+								{ name:"IndianRupee", text:"&#8377;"},];
+			
+			var menuItems = { 'fonteffects': true,
+							  'texteffects': true,
+							  'aligneffects': true,
+							  'textformats':true,
+							  'actions' : true,
+							  'insertoptions' : true,
+							  'extraeffects' : true,
+							  'advancedoptions' : true,
+							  'screeneffects':true,
+
+							  'fonts'	: { "select":true,
+											"default": "Font",
+											"tooltip": "Fonts",
+											"commandname": "fontName",
+											"custom":null },
+
+							  'styles'	: { "select":true,
+											"default": "Formatting",
+											"tooltip": "Paragraph Format",
+											"commandname": "formatBlock",
+												"custom":null },
+
+							 'font_size': {	"select":true,
+											"default": "Font size",
+											"tooltip": "Font Size",
+											"commandname":"fontSize", 
+											"custom":null },
+
+							  'color'	: { "text":"A",
+											"icon": "fa fa-font", 
+											"tooltip": "Text/Background Color",
+											"commandname":null,
+											"custom":function(button){
+													var editor = $(this);
+													var flag = 0;
+													var paletteCntr   = $('<div/>',{id:"paletteCntr",class:"activeColour", css :{"display":"none","width":"335px"}}).click(function(event){event.stopPropagation();});
+													var paletteDiv    = $('<div/>',{id:"colorpellete"});
+													var palette       = $('<ul />',{id:"color_ui"}).append($('<li />').css({"width":"145px","display":"Block","height":"25px"}).html('<div>Text Color</div>'));
+													var bgPalletteDiv = $('<div/>',{id:"bg_colorpellete"});
+													var bgPallette    = $('<ul />',{id:"bgcolor_ui"}).append($('<li />').css({"width":"145px","display":"Block","height":"25px"}).html('<div>Background Color</div>'));
+													if(editor.data("colorBtn")){
+														flag = 1;
+														editor.data("colorBtn",null);
+													}
+													else
+														editor.data("colorBtn",1);
+													if(flag==0){
+														for (var i = 0; i < colors.length; i++){
+															if(colors[i].hex!=null){
+															    palette.append($('<li />').css('background-color', colors[i].hex).mousedown(function(event){ event.preventDefault();}).click(function(){															
+																	var hexcolor = methods.rgbToHex.apply(this,[$(this).css('background-color')]);
+																	methods.restoreSelection.apply(this);
+																	methods.setStyleWithCSS.apply(this);
+																	document.execCommand('forecolor',false,hexcolor);
+																	$('#paletteCntr').remove();
+
+																	editor.data("colorBtn",null);
+																}));
+
+																bgPallette.append($('<li />').css('background-color', colors[i].hex).mousedown(function(event){ event.preventDefault();}).click(function(){															
+																var hexcolor = methods.rgbToHex.apply(this,[$(this).css('background-color')]);
+																methods.restoreSelection.apply(this);
+																methods.setStyleWithCSS.apply(this);
+																document.execCommand('backColor',false,hexcolor);
+																$('#paletteCntr').remove();
+																editor.data("colorBtn",null);
+																}));
+															}
+															else{
+																palette.append($('<li />').css({"width":"145px","display":"Block","height":"5px"}));
+																bgPallette.append($('<li />').css({"width":"145px","display":"Block","height":"5px"}));
+															}
+														} 
+														palette.appendTo(paletteDiv);
+														bgPallette.appendTo(bgPalletteDiv);
+														paletteDiv.appendTo(paletteCntr);
+														bgPalletteDiv.appendTo(paletteCntr)																												
+														paletteCntr.insertAfter(button);
+														$('#paletteCntr').slideDown('slow');
+													}
+													else 
+														$('#paletteCntr').remove();
+												}},
+							
+							  'bold'	: { "text": "B", 
+											"icon": "fa fa-bold", 
+											"tooltip": "Bold", 
+											"commandname":"bold", 
+											"custom":null },
+
+						      'italics'	: { "text":"I", 
+											"icon":"fa fa-italic", 
+											"tooltip":"Italics", 
+											"commandname":"italic",
+											"custom":null },
+
+						     'underline': { "text":"U", 
+											"icon":"fa fa-underline", 
+											"tooltip":"Underline", 
+											"commandname":"underline",
+											"custom":null },
+											
+						     'strikeout': { "text": "Strikeout", 
+											"icon":"fa fa-strikethrough", 
+											"tooltip": "Strike Through", 
+											"commandname":"strikeThrough", 
+											"custom":null },
+
+						     'ol'		: { "text": "N", 
+											"icon": "fa fa-list-ol", 
+											"tooltip": "Insert/Remove Numbered List", 
+											"commandname":"insertorderedlist", 
+											"custom":null },
+
+						     'ul'		: { "text": "Bullet", 
+											"icon": "fa fa-list-ul", 
+											"tooltip": "Insert/Remove Bulleted List", 
+											"commandname":"insertunorderedlist", 
+											"custom":null },
+
+						     'undo'		: { "text": "undo", 
+											"icon": "fa fa-undo", 
+											"tooltip": "Undo", 
+											"commandname":"undo", 
+											"custom":null },
+
+						     'redo'		: { "text": "redo", 
+											"icon": "fa fa-repeat", 
+											"tooltip": "Redo", 
+											"commandname":"redo", 
+											"custom":null },
+
+						     'l_align'	: { "text": "leftalign", 
+											"icon": "fa fa-align-left", 
+											"tooltip": "Align Left", 
+											"commandname":"justifyleft", 
+											"custom":null },
+
+						     'r_align'	: { "text": "rightalign", 
+											"icon": "fa fa-align-right", 
+											"tooltip": "Align Right", 
+											"commandname":"justifyright", 
+											"custom":null },
+
+						     'c_align'	: { "text": "centeralign", 
+											"icon": "fa fa-align-center", 
+											"tooltip": "Align Center", 
+											"commandname":"justifycenter", 
+											"custom":null },
+
+						     'justify'	: { "text": "justify", 
+											"icon": "fa fa-align-justify", 
+											"tooltip": "Justify", 
+											"commandname":"justifyfull", 
+											"custom":null },
+
+							  'unlink'	: { "text": "Unlink", 
+											"icon": "fa fa-unlink", 
+											"tooltip": "Unlink", 
+											"commandname":"unlink", 
+											"custom":null },
+
+						   'insert_link': { "modal": true,
+						   					"modalId": "InsertLink_" + $(this).attr("id"),
+											"icon":"fa fa-link", 
+											"tooltip": "Insert Link", 
+											"modalHeader": "Insert Hyperlink",
+											"modalBody": $('<div/>',{   class:"form-group"
+																	}).append($('<div/>',{
+																		id :"errMsg_" + $(this).attr("id")
+																	})).append($('<input/>',{
+																		type:"text",
+																		id:"inputText_" + $(this).attr("id"),
+																		class:"form-control form-control-link ",
+																		placeholder:"Text to Display",
+																	})).append($('<input/>',{
+																		type:"text",
+																		id:"inputUrl_" + $(this).attr("id"),
+																		required:true,
+																		class:"form-control form-control-link",
+																		placeholder:"Enter URL"
+																	})),
+											"beforeLoad":function(){ 
+												editorObj = this;
+												var _idSuffix = "_" + this.attr("id");
+												$('#inputText' + _idSuffix);
+												$('#inputUrl' + _idSuffix);
+												$(".alert").alert("close");
+												if($(editorObj).data('currentRange')!=''){ 
+													$('#inputText_' +  _idSuffix).val($(editorObj).data('currentRange'));
+												}
+											},
+											"onSave":function(){
+												var urlPattern = /(http|ftp|https):\/\/[\w-]+(\.[\w-]+)+([\w.,@?^=%&amp;:\/~+#-]*[\w@?^=%&amp;\/~+#-])?/;
+												var _idSuffix = "_" + editorObj.attr("id");
+												var targetText = $('#inputText' + _idSuffix).val();
+												var targetURL  = $('#inputUrl' + _idSuffix).val();
+												var range      = $(editorObj).data('currentRange');
+												if(targetURL ==''){
+													methods.showMessage.apply(editorObj,["errMsg","Please enter url"]);
+													return false;
+												}												
+												if(!targetURL.match(urlPattern)){
+													methods.showMessage.apply(editorObj,["errMsg","Enter valid url"]);
+													return false;
+												}													
+												if(range=='' && targetText==''){ 
+													targetText =targetURL;	
+												}
+												if(navigator.userAgent.match(/MSIE/i) || navigator.userAgent.match(/Windows NT.*Trident\//)){	
+													var targetLink='<a href="'+targetURL+'" target="_blank">'+targetText+'</a>';
+													methods.restoreSelection.apply(editorObj,[targetLink,'html']);
+												}
+												else{
+												    methods.restoreSelection.apply(editorObj, [targetText]);																																		
+													document.execCommand('createLink',false,targetURL);
+												}
+												$(editorObj).data("editor").find('a[href="'+targetURL+'"]').each(function(){ $(this).attr("target", "_blank"); });
+												$(".alert").alert("close");
+												$("#InsertLink" + _idSuffix).modal("hide");
+												$(editorObj).data("editor").focus();
+												return false;
+											}},
+
+						   'insert_img'	: { "modal": true,
+						   					"modalId": "InsertImage_" + $(this).attr("id"), 
+											"icon":"fa fa-picture-o", 
+											"tooltip": "Insert Image", 
+											"modalHeader": "Insert Image",
+											"modalBody": methods.imageWidget.apply(this),
+											"beforeLoad":function(){
+												editorObj = this;
+												var _idSuffix = editorObj.attr("id");
+												$('#imageURL_' + _idSuffix).val("");
+												$("#uploadImageBar_" + _idSuffix + " :input").val("");
+												$('#imageList_' + _idSuffix).data('current',"");																																				
+											},
+											"onSave": function(){
+												var _idSuffix = "_" + editorObj.attr("id");
+												methods.restoreSelection.apply(this);												
+												if($('#imageList' + _idSuffix).data('current')){
+													if(navigator.userAgent.match(/MSIE/i) || navigator.userAgent.match(/Windows NT.*Trident\//)){
+														var imageStr = '<img src="'+$('#imageList' + _idSuffix).data('current')+'"/>'
+														methods.restoreSelection.apply(this,[imageStr,'html'])
+													}
+													else{
+														document.execCommand('insertimage', false, $('#imageList' + _idSuffix).data('current'));
+													}
+												}
+												else{
+													methods.showMessage.apply(this,["imgErrMsg" + _idSuffix,"Please select an image"]);
+													return false;
+												}
+												$("#InsertImage" + _idSuffix).modal("hide");
+												$(this).data("editor").focus();
+											}},
+
+						'insert_table'	: { "modal": true,
+					   						"modalId": "InsertTable_" + $(this).attr("id"), 
+											"icon":"fa fa-table", 
+											"tooltip": "Insert Table", 
+											"modalHeader": "Insert Table",
+											"modalBody":methods.tableWidget.apply(this),
+											"beforeLoad":function(){
+												editorObj = this;
+												$('#tblForm_' + editorObj.attr("id")).each (function(){ this.reset(); });																																	
+											},
+											"onSave": function(){
+												_idSuffix = "_" + editorObj.attr("id");
+												methods.restoreSelection.apply(this);
+												var tblRows        = $('#tblRows' + _idSuffix).val();
+												var tblColumns     = $('#tblColumns' + _idSuffix).val();
+												var tblWidth       = $('#tblWidth' + _idSuffix).val();
+												var tblHeight      = $('#tblHeight' + _idSuffix).val();
+												var tblAlign       = $('#tblAlign'  + _idSuffix).val();
+												var tblBorder      = $('#tblBorder' + _idSuffix).val();
+												var tblCellspacing = $('#tblCellspacing' + _idSuffix).val();
+												var tblCellpadding = $('#tblCellpadding' + _idSuffix).val();
+												var intReg 		   = /^[0-9]+$/;
+												var cssReg 		   = /^auto$|^[+-]?[0-9]+\.?([0-9]+)?(px|em|ex|%|in|cm|mm|pt|pc)?$/ig;
+												var numReg 		   = /^[0-9]+\.?([0-9])?$/;
+												
+												if(!tblRows.match(intReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Rows must be a positive number"]);
+													return false;
+												}													
+												if(!tblColumns.match(intReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Columns must be a positive number"]);
+													return false;
+												}
+												if(tblWidth!="" && !tblWidth.match(cssReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Please enter positive number with or without a valid CSS measurement unit (px,em,ex,%,in,cm,mm,pt,pc)"]);
+													return false;
+												}
+												if(tblHeight!="" && !tblHeight.match(cssReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Please enter positive number with or without a valid CSS measurement unit (px,em,ex,%,in,cm,mm,pt,pc)"]);
+													return false;
+												}
+												if(tblBorder!="" && !tblBorder.match(numReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Border size must be a positive number"]);
+													return false;
+												}
+												if(tblCellspacing!="" && !tblCellspacing.match(numReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Cell spacing must be a positive number"]);
+													return false;
+												}
+												if(tblCellpadding!="" && !tblCellpadding.match(numReg)){
+													methods.showMessage.apply(this,["tblErrMsg","Cell padding must be a positive number"]);
+													return false;
+												}
+
+												var htmlTableCntr = $('<div/>');
+												var tblAttributes = [	
+																		{attribute:"align",value:tblAlign},
+																		{attribute:"border",value:tblBorder},
+																		{attribute:"cellspacing",value:tblCellspacing},
+																		{attribute:"cellpadding",value:tblCellpadding},
+																		{attribute:"width",value:tblWidth},
+																		{attribute:"height",value:tblHeight},
+																	];
+												var htmlTable     = methods.getHTMLTable.apply(this, [tblRows, tblColumns, tblAttributes]);
+												htmlTable.appendTo(htmlTableCntr);
+												if(navigator.userAgent.match(/MSIE/i) || navigator.userAgent.match(/Windows NT.*Trident\//))
+												methods.restoreSelection.apply(this,[htmlTableCntr.html(),'html']);
+												else
+												document.execCommand('insertHTML', false, htmlTableCntr.html());
+												$("#InsertTable" + _idSuffix).modal("hide");
+												$(this).data("editor").focus();
+											}},
+
+						   'hr_line'	: { "text": "HR", 
+											"icon":"fa fa-minus", 
+											"tooltip": "Horizontal Rule", 
+											"commandname":"insertHorizontalRule", 
+											"custom":null },
+
+						   'block_quote': { "text": "Block Quote", 
+											"icon":"fa fa-quote-right", 
+											"tooltip": "Block Quote", 
+											"commandname":null, 
+											"custom":function(){ 
+												methods.setStyleWithCSS.apply(this);
+												if(navigator.userAgent.match(/MSIE/i) || navigator.userAgent.match(/Windows NT.*Trident\//)){													 
+													document.execCommand('indent', false, null); 	
+												}
+												else{
+													document.execCommand('formatBlock', false, '<blockquote>');
+												}
+											}},						   
+
+						   'indent'		: { "text": "Indent", 
+											"icon":"fa fa-indent", 
+											"tooltip": "Increase Indent", 
+											"commandname":"indent", 
+											"custom":null },
+
+						   'outdent'	: { "text": "Outdent", 
+											"icon":"fa fa-outdent", 
+											"tooltip": "Decrease Indent", 
+											"commandname":"outdent", 
+											"custom":null },
+
+							'print'		: { "text": "Print", 
+											"icon":"fa fa-print", 
+											"tooltip": "Print", 
+											"commandname":null, 
+											"custom":function(){
+											oDoc = $(this).data("editor");
+											var oPrntWin = window.open("","_blank","width=450,height=470,left=400,top=100,menubar=yes,toolbar=no,location=no,scrollbars=yes");
+											oPrntWin.document.open();
+											oPrntWin.document.write("<!doctype html><html><head><title>Print</title></head><body onload=\"print();\">" + oDoc.html() + "</body></html>");
+											oPrntWin.document.close();
+											}},
+
+							'rm_format'	: { "text": "Remove format", 
+											"icon":"fa fa-eraser", 
+											"tooltip": "Remove Formatting", 
+											"commandname":"removeformat", 
+											"custom":null },
+
+							'select_all': { "text": "Select all", 
+											"icon":"fa fa-file-text", 
+											"tooltip": "Select All", 
+											"commandname":null, 
+											"custom":function(){ 
+												document.execCommand("selectall", null, null);												    
+											}},
+
+							'togglescreen':{ "text": "Toggle Screen",
+											 "icon": "fa fa-arrows-alt",
+											 "tooltip": "Toggle Screen",
+											 "commandname":null,
+											 "custom":function(button, parameters){
+												$(this).data("editor").parent().toggleClass('fullscreen');
+												var statusdBarHeight=0;
+												if($(this).data("statusBar").length)
+												{
+													statusdBarHeight = $(this).data("statusBar").height();
+												}
+												if($(this).data("editor").parent().hasClass('fullscreen'))
+													$(this).data("editor").css({"height":$(this).data("editor").parent().height()-($(this).data("menuBar").height()+statusdBarHeight)-13});
+						                        else
+													$(this).data("editor").css({"height":""});
+						                    }},
+
+							'splchars'	: { "text": "S", 
+											"icon": "fa fa-asterisk", 
+											"tooltip": "Insert Special Character", 
+											"commandname":null, 
+											"custom":function(button){
+													methods.restoreIESelection.apply(this);
+													var flag =0;
+													var splCharDiv = $('<div/>',{id:"specialchar", class:"specialCntr", css :{"display":"none"}}).click(function(event) { event.stopPropagation();});
+													var splCharUi  = $('<ul />',{id:"special_ui"});
+													var editor_Content = this; 
+													if($(this).data("editor").data("splcharsBtn")){
+														flag = 1;
+														$(this).data("editor").data("splcharsBtn", null);
+													}
+													else
+														$(this).data("editor").data("splcharsBtn", 1);
+
+													if(flag==0){
+														for (var i = 0; i < specialchars.length; i++){															
+															splCharUi.append($('<li />').html(specialchars[i].text).attr('title',specialchars[i].name).mousedown(function(event){ event.preventDefault();}).click(function(event){
+																if(navigator.userAgent.match(/MSIE/i) || navigator.userAgent.match(/Windows NT.*Trident\//)){
+																	var specCharHtml = $(this).html();
+																	methods.insertTextAtSelection.apply(this,[specCharHtml,'html']);
+																}
+																else{
+																	document.execCommand('insertHTML',false,$(this).html());
+																}
+																$('#specialchar').remove();
+																$(editor_Content).data("editor").data("splcharsBtn", null);
+															}));
+														}														
+														splCharUi.prependTo(splCharDiv);
+														splCharDiv.insertAfter(button)
+														$('#specialchar').slideDown('slow');
+													}
+													else
+														$('#specialchar').remove();
+											}},
+
+							'source'	: { "text": "Source", 
+											"icon":"fa fa-code", 
+											"tooltip": "Source", 
+											"commandname":null, 
+											"custom":function(button, params){ methods.getSource.apply(this, [button, params]) } },
+											"params": {"obj":null},
+										   };
+
+			var menuGroups = {'texteffects' : ['bold', 'italics', 'underline', 'color'],
+							  'aligneffects': ['l_align','c_align', 'r_align', 'justify'],
+							  'textformats': ['indent', 'outdent', 'block_quote', 'ol', 'ul'],
+							  'fonteffects' : ['fonts', 'styles', 'font_size'],
+							  'actions' : ['undo', 'redo'],
+							  'insertoptions' : ['insert_link', 'unlink', 'insert_img', 'insert_table'],
+							  'extraeffects' : ['strikeout', 'hr_line', 'splchars'],
+							  'advancedoptions' : ['print', 'rm_format', 'select_all', 'source'],
+							  'screeneffects' : ['togglescreen']
+							};
+
+			var settings = $.extend({				
+				'texteffects':true,
+				'aligneffects':true,
+				'textformats':true,
+				'fonteffects':true,
+				'actions' : true,
+				'insertoptions' : true,
+				'extraeffects' : true,
+				'advancedoptions' : true,
+				'screeneffects':true,
+				'bold': true,
+				'italics': true,
+				'underline':true,
+				'ol':true,
+				'ul':true,
+				'undo':true,
+				'redo':true,
+				'l_align':true,
+				'r_align':true,
+				'c_align':true,
+				'justify':true,
+				'insert_link':true,
+				'unlink':true,
+				'insert_img':true,
+				'hr_line':true,
+				'block_quote':true,
+				'source':true,
+				'strikeout':true,
+				'indent':true,
+				'outdent':true,
+				'fonts':fonts,
+				'styles':styles,
+				'print':true,
+				'rm_format':true,
+				'status_bar':true,
+				'font_size':fontsizes,
+				'color':colors,
+				'splchars':specialchars,
+				'insert_table':true,
+				'select_all':true,
+				'togglescreen':true
+			},options);
+
+	       	var containerDiv = $("<div/>",{ class : "row-fluid Editor-container" });
+			var $this = $(this).hide();	       	
+	       	$this.after(containerDiv); 
+	       	var menuBar = $( "<div/>",{ id : "menuBarDiv_" + $(this).attr("id"),
+								  		class : "row-fluid line-control-menu-bar"
+							}).prependTo(containerDiv);
+	       	var editor  = $( "<div/>",{	class : "Editor-editor",
+										css : {overflow: "auto"},
+										contenteditable:"true"
+						 	}).appendTo(containerDiv);
+			var statusBar = $("<div/>", {	id : "statusbar_" + $(this).attr("id"),
+											class: "row-fluid line-control-status-bar",
+											unselectable:"on",
+							}).appendTo(containerDiv);
+	       	$(this).data("menuBar", menuBar);
+	       	$(this).data("editor", editor);
+	       	$(this).data("statusBar", statusBar);
+	       	var editor_Content = this;
+	       	if(settings['status_bar']){
+				editor.keyup(function(event){
+					var wordCount = methods.getWordCount.apply(editor_Content);
+					var charCount = methods.getCharCount.apply(editor_Content);
+					$(editor_Content).data("statusBar").html('<div class="label">'+'Words : '+wordCount+'</div>');
+					$(editor_Content).data("statusBar").append('<div class="label">'+'Characters : '+charCount+'</div>');
+            	});
+	        }	        
+	       	
+	       	
+	       	for(var item in menuItems){
+	       		if(!settings[item] ){ //if the display is not set to true for the button in the settings.	       		
+	       			if(settings[item] in menuGroups){
+	       				for(var each in menuGroups[item]){
+	       					settings[each] = false;
+	       				}
+	       			}
+	       			continue;
+	       		}
+	       		if(item in menuGroups){
+	       			var group = $("<div/>",{class:"btn-group"});	       			
+	       			for(var index=0;index<menuGroups[item].length;index++){
+	       				var value = menuGroups[item][index];	       				
+	       				if(settings[value]){
+       						var menuItem = methods.createMenuItem.apply(this,[menuItems[value], settings[value], true]);
+       						group.append(menuItem);
+       					}
+       					settings[value] = false;
+	       			}
+	       			menuBar.append(group);	       				       			
+	       		}
+	       		else{
+	       			var menuItem = methods.createMenuItem.apply(this,[menuItems[item], settings[item],true]);
+	       			menuBar.append(menuItem);
+	       		}	       		
+	       	}
+
+	       	//For contextmenu	       	
+		    $(document.body).mousedown(function(event) {
+		        var target = $(event.target);
+		        if (!target.parents().andSelf().is('#context-menu')) { // Clicked outside
+		            $('#context-menu').remove();
+		        } 
+		        if (!target.parents().andSelf().is('#specialchar') && (target.closest('a').html()!='<i class="fa fa-asterisk"></i>')) { //Clicked outside
+		        	if($("#specialchar").is(':visible'))
+		            {
+						$(editor_Content).data("editor").data("splcharsBtn", null);
+						$('#specialchar').remove();
+		           	}
+		        }
+		        if (!target.parents().andSelf().is('#paletteCntr') && (target.closest('a').html()!='<i class="fa fa-font"></i>')) { //Clicked outside
+		        	if($("#paletteCntr").is(':visible'))
+		            {
+						$(editor_Content).data("editor").data("colorBtn", null);
+						$('#paletteCntr').remove();
+		           	}
+		        }
+		    });
+		    editor.bind("contextmenu", function(e){
+	       		if($('#context-menu').length)
+	       			$('#context-menu').remove();
+	       		var cMenu 	= $('<div/>',{id:"context-menu"
+	       						}).css({position:"absolute", top:e.pageY, left: e.pageX, "z-index":9999
+	       						}).click(function(event){
+								    event.stopPropagation();
+								});
+	       		var cMenuUl = $('<ul/>',{ class:"dropdown-menu on","role":"menu"});
+	       		e.preventDefault();
+	       		if($(e.target).is('a')){
+				methods.createOpenLinkContext.apply(this,[e,cMenuUl]);
+	       			methods.createLinkContext.apply(this,[e,cMenuUl]);
+	       			cMenuUl.appendTo(cMenu);
+	       		    cMenu.appendTo('body');
+	       		}
+	       		else if($(e.target).is('td') || $(e.target).is("th")){
+	       			methods.createTableContext.apply(this,[e,cMenuUl]);
+	       			cMenuUl.appendTo(cMenu);
+	       		    cMenu.appendTo('body');
+	       		}
+	       		else if($(e.target).is('img')){
+	       				       			
+	       			methods.createImageContext.apply(this,[e,cMenuUl]);
+	       			cMenuUl.appendTo(cMenu);
+	       			cMenu.appendTo('body');
+	       		}
+	       	});
+		},
+		createLinkContext: function(event,cMenuUl){
+			var cMenuli = $('<li/>').append($('<a/>',{
+				id:"rem_link",
+				"href":"javascript:void(0)",
+				"text":"RemoveLink"
+			}).click(function(e){
+				return function(){
+				$(e.target).contents().unwrap();
+				$('#context-menu').remove();
+			}}(event)));
+			cMenuli.appendTo(cMenuUl);
+
+		},
+		createOpenLinkContext: function(event,cMenuUl){
+			var cMenulia = $('<li/>').append($('<a/>',{
+				id:"open_link",
+				"text":"OpenLink"
+			}).click(function(e){
+				return function(){
+				window.open(e.target.getAttribute('href'), '_blank');				
+			}}(event)));
+			cMenulia.appendTo(cMenuUl);
+		},
+		createImageContext: function(event,cMenuUl){
+			var cModalId="imgAttribute";
+			var cModalHeader="Image Attributes";
+			var imgModalBody=methods.imageAttributeWidget.apply(this,["edit"]);
+			var onSave = function(){
+				var urlPattern = /(http|ftp|https):\/\/[\w-]+(\.[\w-]+)+([\w.,@?^=%&amp;:\/~+#-]*[\w@?^=%&amp;\/~+#-])?/;
+				var imageAlt = $('#imgAlt').val();
+				var imageTarget = $('#imgTarget').val();
+				if(imageAlt==""){
+					methods.showMessage.apply(this,["imageErrMsg","Please enter image alternative text"]);
+					return false;
+				}
+				if(imageTarget!=""&& !imageTarget.match(urlPattern)){
+					methods.showMessage.apply(this,["imageErrMsg","Please enter valid url"]);
+					return false;
+				}
+				if($("#imgHidden").val()!=""){
+                        var imgId = $("#imgHidden").val();
+	       				$("#"+imgId).attr('alt',imageAlt);
+	       				if(imageTarget!="")
+	       				{
+	       				 if($("#wrap_"+imgId).length)
+	       				 $("#wrap_"+imgId).attr("href",imageTarget);	
+	       				 else
+					     $("#"+imgId).wrap($('<a/>',{ id:"wrap_"+imgId,href:imageTarget,target:"_blank"}));
+					    }
+					    else
+					    {
+					    	if($("#wrap_"+imgId).length)
+					    	$("#"+imgId).unwrap();
+					    }
+	       		}	       		
+				$("#imgAttribute").modal("hide");
+				editorObj.data("editor").focus();
+			};
+			methods.createModal.apply(this,[cModalId,cModalHeader, imgModalBody, onSave]);
+			var modalTrigger = $('<a/>',{	href:"#"+cModalId,
+       										"text":"Image Attributes",
+											"data-toggle":"modal"
+			}).click( function(e){ 
+				return function(){	
+			        $('#context-menu').remove();
+			        var stamp   = (new Date).getTime();			        
+			        $('#imgAlt').val($(e.target).closest("img").attr("alt"));
+			        $('#imgTarget').val('');
+
+			        if(typeof $(e.target).closest("img").attr("id")!=="undefined"){	
+			            var identifier = $(e.target).closest("img").attr("id");		        	
+			        	$('#imgHidden').val(identifier);
+			        	if($('#wrap_'+identifier).length)
+			        		$('#imgTarget').val($('#wrap_'+identifier).attr("href"));
+			        	else
+			        	 	$('#imgTarget').val('');	
+			        }
+			    	else{			    		
+			    		$(e.target).closest("img").attr("id","img_"+stamp)
+			    		$('#imgHidden').val("img_"+stamp);
+			    	}
+					
+			}}(event));
+			cMenuUl.append($('<li/>').append(modalTrigger))
+					.append($('<li/>').append($('<a/>',{text:"Remove Image"}).click( 
+						function(e) { return function(){ 
+								$('#context-menu').remove();
+								$(e.target).closest("img").remove(); 
+						}}(event))));
+		},
+
+		createTableContext: function(event,cMenuUl){
+			var _idSuffix = "_" + editorObj.attr("id") + "_Edt";
+			var modalId="editProperties_" + editorObj.attr("id");
+			$("#" + modalId).remove();
+       		var modalHeader="Table Properties";
+       		var tblModalBody= methods.tableWidget.apply(this,["edit"]);
+       		var onSave = function(){ 
+       			var tblWidthEdt			= $('#tblWidth' + _idSuffix).val();
+       			var tblHeightEdt		= $('#tblHeight' + _idSuffix).val();
+       			var tblBorderEdt		= $('#tblBorder' + _idSuffix).val();
+       			var tblAlignEdt	        = $('#tblAlign' + _idSuffix).val();
+       			var tblCellspacingEdt	= $('#tblCellspacing' + _idSuffix).val();
+       			var tblCellpaddingEdt	= $('#tblCellpadding' + _idSuffix).val();
+				var tblEdtCssReg 		= /^auto$|^[+-]?[0-9]+\.?([0-9]+)?(px|em|ex|%|in|cm|mm|pt|pc)?$/ig;
+				var tblEdtNumReg 		= /^[0-9]+\.?([0-9])?$/;
+				if(tblWidthEdt!="" && !tblWidthEdt.match(tblEdtCssReg)){
+					methods.showMessage.apply(this,["tblErrMsgEdt","Please enter positive number with or without a valid CSS measurement unit (px,em,ex,%,in,cm,mm,pt,pc)"]);
+					return false;
+				}
+				if(tblHeightEdt!="" && !tblHeightEdt.match(tblEdtCssReg)){
+					methods.showMessage.apply(this,["tblErrMsgEdt","Please enter positive number with or without a valid CSS measurement unit (px,em,ex,%,in,cm,mm,pt,pc)"]);
+					return false;
+				}
+				if(tblBorderEdt!="" && !tblBorderEdt.match(tblEdtNumReg)){
+					methods.showMessage.apply(this,["tblErrMsgEdt","Border size must be a positive number"]);
+					return false;
+				}
+				if(tblCellspacingEdt!="" && !tblCellspacingEdt.match(tblEdtNumReg)){
+					methods.showMessage.apply(this,["tblErrMsgEdt","Cell spacing must be a positive number"]);
+					return false;
+				}
+				if(tblCellpaddingEdt!="" && !tblCellpaddingEdt.match(tblEdtNumReg)){
+					methods.showMessage.apply(this,["tblErrMsgEdt","Cell padding must be a positive number"]);
+					return false;
+				}
+				$(event.target).closest('table').css('width',tblWidthEdt);
+				if(tblHeightEdt!="")
+				$(event.target).closest('table').css('height',tblHeightEdt);
+			    $(event.target).closest('table').attr('align',tblAlignEdt);
+			    $(event.target).closest('table').attr('border',tblBorderEdt);
+			    $(event.target).closest('table').attr('cellspacing',tblCellspacingEdt);
+			    $(event.target).closest('table').attr('cellpadding',tblCellpaddingEdt);
+			    $("#" + modalId).modal("hide");
+				editorObj.data("editor").focus();
+       		};
+       		methods.createModal.apply(this,[modalId,modalHeader, tblModalBody, onSave]);
+       		var modalTrigger = $('<a/>',{	href:"#"+modalId,
+       										"text":"Table Properties",
+											"data-toggle":"modal"
+			}).click( function(e){ return function(){
+					var _idSuffix = "_" + editorObj.attr("id") + "_Edt";
+			        $('#context-menu').remove();			
+					$('#tblRows' + _idSuffix).val($(e.target).closest('table').prop('rows').length);			
+				    $('#tblColumns' + _idSuffix).val($(e.target).closest('table').find('tr')[0].cells.length);
+				    $('#tblRows' + _idSuffix).attr('disabled','disabled');   
+				    $('#tblColumns' + _idSuffix).attr('disabled','disabled');
+				    $('#tblWidth' + _idSuffix).val($(e.target).closest('table').get(0).style.width);
+				    $('#tblHeight' + _idSuffix).val($(e.target).closest('table').get(0).style.height);
+				    $('#tblAlign' + _idSuffix).val($(e.target).closest('table').attr("align"));
+				    $('#tblBorder' + _idSuffix).val($(e.target).closest('table').attr("border"));
+				    $('#tblCellspacing' + _idSuffix).val($(e.target).closest('table').attr("cellspacing"));
+				    $('#tblCellpadding' + _idSuffix).val($(e.target).closest('table').attr("cellpadding"));
+
+				    
+			}}(event));
+       		
+			cMenuUl.append($('<li/>',{class:"dropdown-submenu",css:{display:"block"}})
+       						.append($('<a/>',{"tabindex":"-1", href:"javascript:void(0)","text":"Row"}))
+       						.append($('<ul/>',{class:"dropdown-menu"})
+       								.append($('<li/>').append($('<a/>',{
+											id:"tbl_addrow",
+											"href":"javascript:void(0)",
+											"text":"Add Row"
+											}).click(function(e){
+												return function(){
+													$("#context-menu").remove();
+													var selectedRow = $(e.target).closest("tr");
+													var newRow = $("<tr/>");
+													selectedRow.children().each(function() {
+														var newColumn = $("<" + $(this).prop("nodeName") + "/>").html("&nbsp;");
+														newRow.append(newColumn);
+													});
+													selectedRow.after(newRow);
+												}
+											}(event))))
+       								.append($('<li/>').append($('<a/>',{text:"Remove Row"}).click( 
+											function(e) { return function(){ 
+													$('#context-menu').remove();
+													$(e.target).closest("tr").remove(); 
+											}}(event))))
+       			)).append($('<li/>',{class:"dropdown-submenu",css:{display:"block"}})
+   						.append($('<a/>',{"tabindex":"-1", href:"javascript:void(0)","text":"Column"}))
+   						.append($('<ul/>',{class:"dropdown-menu"})
+   								.append($('<li/>').append($('<a/>',{
+										id:"tbl_addcolumn",
+										"href":"javascript:void(0)",
+										"text":"Add Column",
+										}).click(function(e){
+											return function(){
+												$('#context-menu').remove();
+												var selectedCell = $(e.target);
+												var columnIndex = selectedCell.siblings().addBack().index(selectedCell);
+												selectedCell.closest("table").find("tr").each(function() {
+													var cellInSelectedColumn = $(this).children(":eq(" + columnIndex + ")");
+													var newCell = $("<" + cellInSelectedColumn.prop("nodeName") + "/>").html("&nbsp;");
+													cellInSelectedColumn.after(newCell);
+												});
+											}
+										}(event))))
+   								.append($('<li/>').append($('<a/>',{text:"Remove Column"}).click( 
+										function(e) { return function(){ 
+												$('#context-menu').remove();
+												var selectedCell = $(e.target);
+												var columnIndex = selectedCell.siblings().addBack().index(selectedCell);
+												selectedCell.closest("table").find("tr").each(function() {
+													$(this).children(":eq(" + columnIndex + ")").remove();
+												});
+										}}(event))))
+   						));
+			cMenuUl.append($('<li/>').append(modalTrigger))
+					.append($('<li/>',{class:"divider"}))
+					.append($('<li/>').append($('<a/>',{text:"Remove Table"}).click( 
+						function(e){ return function(){ 
+								$('#context-menu').remove();
+								$(e.target).closest("table").remove(); 
+						}}(event))));
+
+		},
+
+		createModal: function(modalId, modalHeader, modalBody, onSave){
+			//Create a Modal for the button.		
+			var modalTrigger = $('<a/>',{	href:"#"+modalId,
+											role:"button",
+											class:"btn btn-default",
+											"data-toggle":"modal"
+			});
+			var modalElement = $('<div/>',{ id: modalId,
+								           class: "modal fade",
+								              tabindex: "-1",
+								              role: "dialog",
+								              "aria-labelledby":"h3_"+modalId,
+								              "aria-hidden":"true"
+								          }).append($('<div>',{
+								            	class:"modal-dialog"
+								         		}).append($('<div>',{
+							            			class:"modal-content"
+									         		}).append($('<div>',{
+									           			class:"modal-header"
+									           			}).append($('<button/>',{
+										                	type:"button",
+										                	class:"close",
+										                	"data-dismiss":"modal",
+										                	"aria-hidden":"true"
+										               		}).html('x')
+									            		).append($('<h3/>',{
+									                		id:"h3_"+modalId
+									           				}).html(modalHeader))
+									         		).append($('<div>',{
+									           			class:"modal-body"
+									           			}).append(modalBody)
+									          		).append($('<div>',{
+									            		class:"modal-footer"
+									         			}).append($('<button/>',{
+									                		type:"button",
+									                		class:"btn btn-default",
+									                		"data-dismiss":"modal",
+									                		"aria-hidden":"true"
+									               			}).html('Cancel')
+								           	  			).append($('<button/>',{
+								                			type:"button",
+								                			class:"btn btn-success",
+								               				}).html('Done').mousedown(function(e){
+								                			e.preventDefault();
+								               				}).click(function(obj){return function(){onSave.apply(obj)}}(this)))
+	         								  		)
+       											)	
+       									);	
+			modalElement.appendTo("body");
+			return modalTrigger;
+		},
+
+		createMenuItem: function(itemSettings, options, returnElement){
+			//Function to perform multiple actions.supplied arguments: itemsettings-list of buttons and button options, options: options for select input, returnelement: boolean.
+			//1.Create Select Options using Bootstrap Dropdown.
+			//2.Create modal dialog using bootstrap options
+			//3.Create menubar buttons binded with corresponding event actions
+			typeof returnElement !== 'undefined' ? returnElement : false;
+
+			if(itemSettings["select"]){
+				var menuWrapElement = $("<div/>", {class:"btn-group"});
+				var menuElement 	= $("<ul/>", {class:"dropdown-menu"});
+				menuWrapElement.append($('<a/>',{
+										class:"btn btn-default dropdown-toggle",
+										"data-toggle":"dropdown",
+										"href":"javascript:void(0)",
+										"title":itemSettings["tooltip"]
+										}).html(itemSettings["default"]).append($("<span/>",{class:"caret"})).mousedown(function(e){
+											e.preventDefault();
+										}));
+				$.each(options,function(i,v){
+					var option = $('<li/>')
+		            $("<a/>",{
+		              tabindex : "-1",
+		              href : "javascript:void(0)"
+		            }).html(i).appendTo(option);
+
+		            option.click(function(){
+		            	$(this).parent().parent().data("value", v);
+		            	$(this).parent().parent().trigger("change")
+		            });
+		            menuElement.append(option);		            
+		        });
+				var action = "change";
+		    }
+		    else if(itemSettings["modal"]){
+		    	var menuWrapElement = methods.createModal.apply(this,[itemSettings["modalId"], itemSettings["modalHeader"], itemSettings["modalBody"], itemSettings["onSave"]]);		    			    	
+		    	var menuElement = $("<i/>");
+		    	if(itemSettings["icon"])
+					menuElement.addClass(itemSettings["icon"]);
+				else
+					menuElement.html(itemSettings["text"]);
+				menuWrapElement.append(menuElement);
+				menuWrapElement.mousedown(function(obj, methods, beforeLoad){
+					return function(e){
+						e.preventDefault();
+						methods.saveSelection.apply(obj);
+						if(beforeLoad){		    	    
+							beforeLoad.apply(obj); 					
+				    	}
+					}
+				}(this, methods,itemSettings["beforeLoad"]));
+				menuWrapElement.attr('title', itemSettings['tooltip']);
+				return menuWrapElement;
+		    }
+			else{
+				var menuWrapElement = $("<a/>",{href:'javascript:void(0)', class:'btn btn-default'});
+				var menuElement = $("<i/>");
+				if(itemSettings["icon"])
+					menuElement.addClass(itemSettings["icon"]);
+				else
+					menuElement.html(itemSettings["text"]);
+				var action = "click";
+			}
+			if(itemSettings["custom"]){
+				menuWrapElement.bind(action, (function(obj, params){
+						return function(){
+						methods.saveSelection.apply(obj);
+						itemSettings["custom"].apply(obj, [$(this), params]);
+						}
+					})(this, itemSettings['params']));
+			}
+			else{
+				menuWrapElement.data("commandName", itemSettings["commandname"]);
+				menuWrapElement.data("editor", $(this).data("editor"));
+				menuWrapElement.bind(action, function(){ methods.setTextFormat.apply(this) });
+			}
+			menuWrapElement.attr('title', itemSettings['tooltip']);
+			menuWrapElement.css('cursor', 'pointer');
+			menuWrapElement.append(menuElement);
+			if(returnElement)
+				return menuWrapElement;
+			$(this).data("menuBar").append(menuWrapElement);
+		},
+
+		setTextFormat: function(){			
+			//Function to run the text formatting options using execCommand.
+			methods.setStyleWithCSS.apply(this);
+			document.execCommand($(this).data("commandName"), false, $(this).data("value") || null);
+			$(this).data("editor").focus();
+			return false;
+		},
+
+		getSource: function(button, params){
+			//Function to show the html source code to the editor and toggle the text display.
+			var flag = 0;
+			if(button.data('state')){
+				flag = 1;
+				button.data('state', null);
+			}
+			else
+				button.data('state', 1);
+			$(this).data("source-mode", !flag);
+			var editor = $(this).data('editor');
+			var content;
+			if(flag==0){ //Convert text to HTML			
+				content = document.createTextNode(editor.html());
+				editor.empty();
+				editor.attr('contenteditable', false);
+				preElement = $("<pre/>",{
+					contenteditable: true					
+					});
+				preElement.append(content);				
+				editor.append(preElement);
+				button.parent().siblings().hide();
+				button.siblings().hide();
+			}
+			else{
+				var html = editor.children().first().text();
+				editor.html(html);
+				editor.attr('contenteditable', true);
+				button.parent().siblings().show();
+				button.siblings().show();
+			}
+		},
+
+		countWords: function(node){
+			//Function to count the number of words recursively as the text grows in the editor.
+			var count = 0;	
+    		var textNodes = node.contents().filter(function() { 
+				return (this.nodeType == 3); 
+			});			
+			for(var index=0;index<textNodes.length;index++){
+				text = textNodes[index].textContent;
+				text = text.replace(/[^-\w\s]/gi, ' ');
+				text = $.trim(text);
+				count = count + text.split(/\s+/).length;
+			}
+			var childNodes = node.children().each(function(){
+				count = count + methods.countWords.apply(this, [$(this)]);
+			});
+			return count
+		},
+
+		countChars: function(node){
+			//Function to count the number of characters recursively as the text grows in the editor.
+			var count = 0;
+    		var textNodes = node.contents().filter(function() { 
+				return (this.nodeType == 3); 
+			});
+			for(var index=0;index<textNodes.length;index++){
+				text = textNodes[index].textContent;
+				count = count + text.length;
+			}
+			var childNodes = node.children().each(function(){
+				count = count + methods.countChars.apply(this, [$(this)]);
+			});
+			return count;
+		},
+
+		getWordCount: function(){
+			//Function to return the word count of the text in the editor
+			return methods.countWords.apply(this, [$(this).data("editor")]);
+		},
+
+		getCharCount: function(){
+			//Function to return the character count of the text in the editor
+			return methods.countChars.apply(this, [$(this).data("editor")]);
+		},
+
+		rgbToHex: function(rgb){
+			//Function to convert the rgb color codes into hexadecimal code
+			rgb = rgb.match(/^rgb\((\d+),\s*(\d+),\s*(\d+)\)$/);
+			return "#" +
+			("0" + parseInt(rgb[1],10).toString(16)).slice(-2) +
+			("0" + parseInt(rgb[2],10).toString(16)).slice(-2) +
+			("0" + parseInt(rgb[3],10).toString(16)).slice(-2);
+		},
+
+		showMessage: function(target,message){
+			//Function to show the error message. Supplied arguments:target-div id, message-message text to be displayed.
+			var errorDiv=$('<div/>',{ class:"alert alert-danger"	}
+				).append($('<button/>',{
+									type:"button",
+									class:"close",
+									"data-dismiss":"alert",
+									html:"x"
+				})).append($('<span/>').html(message));
+			errorDiv.appendTo($('#'+target));
+			setTimeout(function() { $('.alert').alert('close'); }, 3000);								
+		},
+		
+		getText: function(){
+			//Function to get the source code.
+			if(!$(this).data("source-mode"))
+				return $(this).data("editor").html();
+			else
+				return $(this).data("editor").children().first().text();
+		},
+
+		setText: function(text){
+			//Function to set the source code
+			if(!$(this).data("source-mode"))
+				$(this).data("editor").html(text);
+			else
+				$(this).data("editor").children().first().text(text);
+		},
+
+		setStyleWithCSS:function(){
+			if(navigator.userAgent.match(/MSIE/i)){	//for IE10
+				try {
+                	Editor.execCommand("styleWithCSS", 0, false);
+            	} catch (e) {
+	                try {
+	                    Editor.execCommand("useCSS", 0, true);
+	                } catch (e) {
+	                    try {
+	                        Editor.execCommand('styleWithCSS', false, false);
+	                    }
+	                    catch (e) {
+	                    }
+	                }
+            	}
+			}
+			else{
+				document.execCommand("styleWithCSS", null, true);
+			}
+		},				
+
+	}
+
+	$.fn.Editor = function( method ){
+
+		if ( methods[method] ) {
+			return methods[method].apply( this, Array.prototype.slice.call( arguments, 1 ));
+		} else if ( typeof method === 'object' || ! method ) {
+			return methods.init.apply( this, arguments );
+		} else {
+			$.error( 'Method ' +  method + ' does not exist on jQuery.Editor' );
+		}    
+	}; 
+})( jQuery );

Diff do ficheiro suprimidas por serem muito extensas
+ 23 - 0
backstage/static/js/jquery.richtext.min.js


+ 3 - 4
backstage/static/js/yo.js

@@ -1,3 +1,6 @@
+const contentApiUrl = `${PORTAL_SERVER}contents?url=`;
+frontMatters = [];
+contentMatters = [];
 
 var SwfType = {};
 if (location.pathname.indexOf('system_furniture') >= 0) {
@@ -33,9 +36,6 @@ $(function () {
 
 });
 
-frontMatters = [];
-contentMatters = [];
-
 function getHeader(url) {
     axios.get(contentApiUrl + url).then(({ data }) => {
         frontMatters = [];
@@ -392,7 +392,6 @@ editorBlocks1 = [];
 function ParseProductSection(inContent) {
     editorBlocks = [];
     editorBlocks1 = [];
-
     var aa = $.parseHTML(inContent.trim());
     //處理圖片
     if ($("[id='carousel-with-preview']", aa).length > 0) {

+ 322 - 0
backstage/static/styles/htmleditor.css

@@ -0,0 +1,322 @@
+/*!
+ * http://suyati.github.io/line-control
+ * LineControl 1.1.0
+ * Copyright (C) 2014, Suyati Technologies
+ * This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version.
+This program 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 General Public License for more details.
+You should have received a copy of the GNU General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ *
+*/
+
+@charset "utf-8";
+/* base styles */
+.jumbotron {
+   background-color: #67A754;
+  background-image: url(images/radial_bg.png);
+  background-position: center center;
+  background-repeat: no-repeat; 
+  background: -webkit-gradient(radial, center center, 0, center center, 460, from(#B3ECFF), to(#7EDEFF)); /* Safari 4-5, Chrome 1-9 */ /* Can't specify a percentage size? Laaaaaame. */
+  background: -webkit-radial-gradient(circle, #B3ECFF, #7EDEFF); /* Safari 5.1+, Chrome 10+ */
+  background: -moz-radial-gradient(circle, #B3ECFF, #7EDEFF); /* Firefox 3.6+ */
+  background: -ms-radial-gradient(circle, #B3ECFF, #7EDEFF); /* IE 10 */
+    box-shadow: 0 3px 7px rgba(0, 0, 0, 0.2) inset, 0 -3px 7px rgba(0, 0, 0, 0.2) inset;
+    color: #FFFFFF;
+    padding: 40px 0;
+    position: relative;
+    text-shadow: 0 1px 3px rgba(255, 255, 255, 0.4), 0 0 30px rgba(255, 255, 255, 0.075);
+}
+.jumbotron h1{font-family: 'Roboto', sans-serif!important; font-size:45px;font-weight:300; border-bottom:1px solid rgba(255,255,255,0.4);padding-bottom:20px; }
+.jumbotron h1 span{color:rgba(0,0,0,0.4); }
+.jumbotron p {
+  font-family: 'Roboto', sans-serif!important;
+    font-size: 30px;
+    font-weight: 300;
+    line-height: 1.25;
+  color:#444;
+}
+
+.jumbotron .container {
+    position: relative;
+    z-index: 2;
+}
+.jumbotron:after {
+    background: url("../images/pattern.png") repeat scroll center center transparent;
+    bottom: 0;
+    content: "";
+    display: block;
+    left: 0;
+    opacity: 0.4;
+    position: absolute;
+    right: 0;
+    top: 0;
+}
+.jumbotron:after {
+    background-size: 400px 400px;
+}
+.masthead {
+    color: #FFFFFF;
+    margin-bottom: 0;
+    padding: 30px 0 10px;
+}
+h2.demo-text{font-family: 'Roboto', sans-serif!important; font-size:45px; font-weight:300; color:#51D2FF; text-align:center; margin:20px 0; line-height:40px;}
+.features{padding:20px 0 10px 0;background:#EEE;font-family: 'Roboto', sans-serif!important; font-size:25px; line-height:35px; text-align:center;font-weight:300;}
+.footer{border-top:1px solid #DDD; margin:20px 0 10px 0; padding-top:10px; font-size:12px;}
+
+/* Editor Styles */
+
+.Editor-container{
+  margin-top:10px;
+  font-family:Gotham, "Helvetica Neue", Helvetica, Arial, sans-serif;
+  }
+  
+.line-control-menu-bar{
+   background-color: #f5f5f5;
+  *background-color: #e6e6e6;
+  background-image: -moz-linear-gradient(top, #ffffff, #e6e6e6);
+  background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#ffffff), to(#e6e6e6));
+  background-image: -webkit-linear-gradient(top, #ffffff, #e6e6e6);
+  background-image: -o-linear-gradient(top, #ffffff, #e6e6e6);
+  background-image: linear-gradient(to bottom, #ffffff, #e6e6e6);
+  background-repeat: repeat-x;
+  border: 1px solid #cccccc;
+  *border: 0;
+  border-color: #e6e6e6 #e6e6e6 #bfbfbf;
+  border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
+  border-bottom-color: #b3b3b3;
+  filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#ffe6e6e6', GradientType=0);
+  filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
+  *zoom: 1;
+      }
+  .line-control-status-bar{
+   background-color: #f5f5f5;
+  *background-color: #e6e6e6;
+  background-image: -moz-linear-gradient(top, #ffffff, #e6e6e6);
+  background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#ffffff), to(#e6e6e6));
+  background-image: -webkit-linear-gradient(top, #ffffff, #e6e6e6);
+  background-image: -o-linear-gradient(top, #ffffff, #e6e6e6);
+  background-image: linear-gradient(to bottom, #ffffff, #e6e6e6);
+  background-repeat: repeat-x;
+  border: 1px solid #cccccc;
+  *border: 0;
+  border-color: #e6e6e6 #e6e6e6 #bfbfbf;
+  border-color: rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.1) rgba(0, 0, 0, 0.25);
+  border-bottom-color: #b3b3b3;
+  filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#ffe6e6e6', GradientType=0);
+  filter: progid:DXImageTransform.Microsoft.gradient(enabled=false);
+  *zoom: 1;
+      }
+  
+.line-control-menu-bar{
+  border-radius:3px 3px 0 0;
+  border-bottom:none;
+  min-height:25px;
+  padding:5px 0;
+  }
+.line-control-menu-bar div:first-child{margin-left:3px;} 
+.line-control-menu-bar .dropdown{float:left;}
+.line-control-menu-bar .dropdown-toggle{font-size:14px; }
+.line-control-menu-bar .dropdown-toggle .caret{margin-left:5px; border-top-color:#999;}
+.line-control-menu-bar a.btn{position:relative; margin-bottom:3px;}
+#paletteCntr{}
+select, textarea, input[type="text"], input[type="password"], input[type="datetime"], input[type="datetime-local"], input[type="date"], input[type="month"], input[type="time"], input[type="week"], input[type="number"], input[type="email"], input[type="url"], input[type="search"], input[type="tel"], input[type="color"], .uneditable-input{height:auto; margin-bottom:10px;}
+  
+.line-control-status-bar{
+  min-height:30px;
+  border-radius:0 0 3px  3px;
+  border-top:none;
+  line-height:30px;
+  text-shadow: 0 1px 1px rgba(255, 255, 255, 0.75);
+  }
+.line-control-status-bar .label{
+  float:right; 
+  margin:6px 10px 0 0; 
+  background:#BBB; 
+  text-shadow: 0px -1px 0px rgba(0, 0, 0, 0.25); 
+  padding: 2px 4px;
+  font-size: 11.844px;
+  line-height:14px;
+} 
+.Editor-editor{ 
+    height:300px;
+    padding:1%;  
+  border:1px solid #EEE;
+  border-radius:0;
+    word-wrap: break-word;
+  }
+#paletteCntr ul
+{    
+    width: 145px;
+    float: left;
+    z-index: 999;
+  margin:0 10px;
+}
+#paletteCntr li
+{
+    cursor: pointer;
+    display: block;
+    height: 16px;
+    float: left;
+    margin: 0;
+    padding: 0;
+    width: 16px;
+    margin:1px;
+}
+#paletteCntr li div{text-align:left; margin:0; font-size:12px;}
+div.activeColour
+{
+    position: absolute;
+    display: inline;
+    left: 0;
+    top:30px;
+    background:#E8E8E8;
+    padding-bottom:10px;
+    z-index:10000;
+          
+}
+#colorpellete, #bg_colorpellete{float:left;}
+
+.bg_activeColour
+{
+    position: absolute;
+    display: inline;
+    float: left;
+}
+
+.specialCntr
+{
+    background-color:#E8E8E8!important;
+    display:inline;
+    position:absolute;
+  width:176px;
+  top:30px;
+  left:0;
+}
+.specialCntr ul{float:left; margin:10px; white-space:normal; -webkit-padding-start: 0px; -moz-padding-start: 0px;}
+.specialCntr ul li{list-style:none; float:left; font-size:12px; width:20px; height:20px; background:#FFF; margin:3px; color:#333; text-align:center; font-weight:bold; cursor:pointer;}
+.specialCntr ul li:hover{background:#666; color:#FFF;}
+pre {
+  white-space: pre-wrap;       /* css-3 */
+  white-space: -moz-pre-wrap;  /* Mozilla, since 1999 */
+  white-space: -pre-wrap;      /* Opera 4-6 */
+  white-space: -o-pre-wrap;    /* Opera 7 */
+  word-wrap: break-word;       /* Internet Explorer 5.5+ */
+}
+.on{display:block; position:relative;}
+.fullscreen{position:fixed; background:#FFF; width:100%; height:100%; top:0; left:0; z-index:1000; margin:0;}
+.fullscreen .line-control-status-bar{position:fixed; bottom:0; border-top:1px solid #DDD;}
+.fullscreen #contentarea{border:none;}
+
+/***Custome Added Css for Left Tabs***/
+.tabs-below > .nav-tabs,
+.tabs-right > .nav-tabs,
+.tabs-left > .nav-tabs {
+  border-bottom: 0;
+}
+.tabs-left ul{
+  -webkit-padding-start: 0px;
+  }
+.tabs-left > .nav-tabs > li,
+.tabs-right > .nav-tabs > li {
+  float: none;
+}
+.tabs-left > .nav-tabs > li > a,
+.tabs-right > .nav-tabs > li > a {
+  min-width: 74px;
+  margin-right: 0;
+  margin-bottom: 3px;
+}
+.tabs-left > .nav-tabs {
+  float: left;
+  margin-right: 19px;
+  border-right: 1px solid #ddd;
+}
+.tabs-left > .nav-tabs > li > a {
+  margin-right: -1px;
+  -webkit-border-radius: 4px 0 0 4px;
+     -moz-border-radius: 4px 0 0 4px;
+          border-radius: 4px 0 0 4px;
+}
+.tabs-left > .nav-tabs > li > a:hover,
+.tabs-left > .nav-tabs > li > a:focus {
+  border-color: #eeeeee #dddddd #eeeeee #eeeeee;
+  outline:none;
+}
+.tabs-left > .nav-tabs .active > a,
+.tabs-left > .nav-tabs .active > a:hover,
+.tabs-left > .nav-tabs .active > a:focus {
+  border-color: #ddd transparent #ddd #ddd;
+  *border-right-color: #ffffff;
+  color:rgb(0, 176, 245) !important;
+  outline:none;
+}
+/**form control width**/
+@-moz-document url-prefix() {
+   .form-control{
+  padding:6px 12px 10px 12px;
+  }
+}
+.form-control-width{
+  width:100px;
+  }
+.form-control-link{
+  width:204px;
+  display:inline-block;
+  margin-bottom:5px;
+  margin-right:5px;
+  }
+.form-control-button-right{
+  width:204px;
+  }
+.inline-form-control{
+  display:inline !important;
+  }
+
+.btn-group + .btn-group {
+  margin-left: 5px;
+}
+.activeColour ul{
+  -webkit-padding-start: 0px;
+  -moz-padding-start: 0px;
+  }
+h1{
+  margin-top:10px;
+  line-height:40px;
+  }
+.col-lg-6 nth:child(1){
+  margin-left:0px;
+  }
+.nopadding{
+  padding:0px;
+  }
+.nopadding-right{
+  padding-right:0px;
+  }
+.modal{
+  overflow-y:auto;
+  }
+.padding-top{
+  padding-top:10px;
+  }
+/*** Drop Down Menu***/
+.dropdown-submenu
+{position:relative;}
+.dropdown-submenu>.dropdown-menu{top:0;left:100%;margin-top:-6px;margin-left:-1px;-webkit-border-radius:0 6px 6px 6px;-moz-border-radius:0 6px 6px 6px;border-radius:0 6px 6px 6px;}
+.dropdown-submenu:hover>.dropdown-menu{display:block;}
+.dropdown-submenu>a:after{display:block;content:" ";float:right;width:0;height:0;border-color:transparent;border-style:solid;border-width:5px 0 5px 5px;border-left-color:#cccccc;margin-top:5px;margin-right:-10px;}
+.dropdown-submenu:hover>a:after{border-left-color:#ffffff;}
+.dropdown-submenu.pull-left{float:none;}.dropdown-submenu.pull-left>.dropdown-menu{left:-100%;margin-left:10px;-webkit-border-radius:6px 0 6px 6px;-moz-border-radius:6px 0 6px 6px;border-radius:6px 0 6px 6px;}
+.dropdown-menu>li>a:hover, .dropdown-menu>li>a:focus, .dropdown-submenu:hover>a, .dropdown-submenu:focus>a {
+  color: #fff;
+  text-decoration: none;
+  background-color: #0081c2;
+  background-image: -moz-linear-gradient(top,#08c,#0077b3);
+  background-image: -webkit-gradient(linear,0 0,0 100%,from(#08c),to(#0077b3));
+  background-image: -webkit-linear-gradient(top,#08c,#0077b3);
+  background-image: -o-linear-gradient(top,#08c,#0077b3);
+  background-image: linear-gradient(to bottom,#08c,#0077b3);
+  background-repeat: repeat-x;
+  filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff0088cc',endColorstr='#ff0077b3',GradientType=0);
+}
+.dropdown-menu>li>a{
+  cursor:pointer;
+  }

Diff do ficheiro suprimidas por serem muito extensas
+ 22 - 0
backstage/static/styles/richtext.min.css


+ 33 - 182
backstage/templates/contact_us.html

@@ -1,191 +1,42 @@
 {% extends "layout.html" %}
-{% block table_body %}
+{% block main %}
 <!-- <script type="text/javascript" src="/static/config.js"></script> -->
 <script src="https://cdn.jsdelivr.net/npm/@editorjs/editorjs@latest"></script>
 <script src="https://cdn.jsdelivr.net/npm/@editorjs/image@latest"></script><!-- Image -->
 <script type="text/javascript" src="/static/js/yo.js"></script>
-<tbody id='contactData'>
-
-    <tr>
-        <td class="table__data"></td>
-        <td class="table__data"></td>
-        <td class="table__data"></td>
-        <!-- <td class="table__data">1</td> -->
-        <td class="table__data"></td>
-        <td>
-            <div class="d-flex justify-content-center">
-                <button class="btn btn_light mr-1" onclick="getHeader('');"><b>修改</b> <i
-                        class="fas fa-pencil-alt"></i></button>
-            </div>
-
-        </td>
-    </tr>
-
-</tbody>
-
-{% endblock table_body %}
-
-
-{% block modal_body %}
-
-<div class="form-group">
-    類別:<select id="newSwfDropdown"></select><br />
-    名稱:<input id="newSwfName" type="text" /><br />
-    <!--     敘述:<input id="swfDesc" type="text" /><br />
-    <hr>
-    圖片:
-    <div id="editorjs" style='border:inset 1px;'></div>
-    <hr>
-    定價:<input id="swfPrice" type="text" /><br />
-    顏色:<input id="swfColor" type="text" /><br />
-    尺寸:<input id="swfSize" type="text" /><br />
-    材質:<input id="swfMet" type="text" /><br /> -->
-</div>
-<div class="modal-footer  pb-0 border-0">
-    <button type="button" class="btn btn__cancel" data-dismiss="modal">取消</button>
-    <button type="button" class="btn btn__submitadd" onclick="newfur();">完成</button>
-</div>
-
-{% endblock modal_body %}
-
-{% block main_info_modal_body %}
-<style>
-    .image-tool__caption {
-        display: none;
-    }
-</style>
-<div class="modal fade" id="myModal">
-    <div class="modal-dialog">
-        <div class="modal-content">
-
-            <div class="modal-header">
-                <h4 class="modal-title">單品家具-修改</h4>
-                <button type="button" class="close" data-dismiss="modal">×</button>
-            </div>
-
-            <div class="modal-body">
-                <table class="table table-bordered">
-                    <tbody>
-                        <tr>
-                            <td>
-                                <h4>名稱</h4>
-                            </td>
-                            <td>
-                                <input class="form-control" id="ctitle" type="text" />
-                                <div class="mt-1 text-danger">(建議字數: 15字內)</div>
-                            </td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>類別</h4>
-                            </td>
-                            <td><select id="swfDropdown" onchange="$('#ctype').val(this.value);" disabled></select></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>敘述</h4>
-                            </td>
-                            <td>
-                                <textarea class="form-control" rows="3" id="swfDesc" type="text"
-                                    style="width: 100%;"></textarea>
-                                <div class="mt-1 text-danger">(建議字數: 50字內)</div>
-                            </td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>日期</h4>
-                            </td>
-                            <td><input class="form-control" id="cdate" type="text" /></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>是否顯示</h4>
-                            </td>
-                            <td><input id="cdraft" type="checkbox" checked="true" /></td>
-                        </tr>
-                        <tr>
-                            <td colspan="2"> </td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>單品圖</h4>
-                            </td>
-                            <td>
-                                <div class="form-control" id="editorjs" style='border:inset 1px;'></div>
-                            </td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>定價</h4>
-                            </td>
-                            <td><input class="form-control" id="swfPrice" type="text" style="width: 100%;" /></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>顏色</h4>
-                            </td>
-                            <td><input class="form-control" id="swfColor" type="text" style="width: 100%;" /></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>尺寸(mm)</h4>
-                            </td>
-                            <td><input class="form-control" id="swfSize" type="text" style="width: 100%;" /></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>材質</h4>
-                            </td>
-                            <td><input class="form-control" id="swfMat" type="text" style="width: 100%;" /></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>備註</h4>
-                            </td>
-                            <td><input class="form-control" id="swfMemo" type="text" style="width: 100%;" /></td>
-                        </tr>
-                        <tr>
-                            <td>
-                                <h4>規格圖</h4>
-                            </td>
-                            <td>
-                                <div class="form-control" id="editorjs1" style='border:inset 1px;'></div>
-                            </td>
-                        </tr>
-                    </tbody>
-                </table>
-            </div>
-
-            <div class="modal-footer">
-                <button type="button" class="btn btn__cancel" data-dismiss="modal">取消</button>
-                <button type="button" class="btn btn__submitadd" id='addNewButton'
-                    onclick="UpdateProduct();">完成</button>
-                <!-- <button type="button" id="uptbtn" class="btn btn-danger" onclick="updateHeader();">完成修改</button> -->
-            </div>
-
-        </div>
-    </div>
-</div>
+<table>
+    <tbody id='contactData'>
+
+        <tr>
+            <td class="table__data">修改聯絡EMAIL</td>
+            <td class="table__data"><input id="newEmail" type="text" style="width: 600px;" /></td>
+            <td class="table__data"></td>
+            <!-- <td class="table__data">1</td> -->
+            <td class="table__data"></td>
+            <td>
+                <div class="d-flex justify-content-center">
+                    <button class="btn btn_light mr-1" onclick="modEmail($('#newEmail').val());"><b>修改</b> <i
+                            class="fas fa-pencil-alt"></i></button>
+                </div>
+
+            </td>
+        </tr>
+
+    </tbody>
+</table>
 <script>
-
-    function UpdateProduct() {
-
-        var mdContent = GetMdHeader();
-
-        mdContent += '\n' + sectionContent;
-        //GenProductSection(outputData);
-        postData = {
-            content: mdContent,
-            url: editTarget
-        };
-
-        /* alert(contentApiUrl);
-        alert(editTarget); */
-        //alert(mdContent);
-        /* axios.post(contentApiUrl + editTarget, json = postData).then(({ data }) => {
+    $(function(){
+        axios.get('/backstage/edit_contact_us_getemail').then(({ data }) => {
+            $('#newEmail').val(data);
+        });
+        
+    });
+
+    function modEmail() {
+        axios.get('/backstage/edit_contact_us_editemail?newemail='+$('#newEmail').val()).then(({ data }) => {
             alert('已儲存');
-        }).finally((data) => { location.reload(); }); */
-
+        }).finally((data) => { location.reload(); });
     };
+
 </script>
-{% endblock %}
+{% endblock main %}

+ 157 - 2
backstage/templates/frequently_asked_questions.html

@@ -1,4 +1,159 @@
 {% extends "layout.html" %}
 {% block main %}
-    <h1>frequently_asked_questions</h1>
-{% endblock main %}
+<script type="text/javascript" src="{{url_for('static', filename='config.js')}}"></script>
+<script type="text/javascript" src="/static/js/yo.js"></script>
+<script src="https://code.jquery.com/jquery-3.6.0.js"></script>
+<script type="text/javascript" src="/static/js/jquery.richtext.min.js"></script>
+<link rel="stylesheet" href="/static/styles/richtext.min.css">
+<script src="https://code.jquery.com/ui/1.13.0/jquery-ui.js"></script>
+<link rel="stylesheet" href="https://code.jquery.com/ui/1.13.0/themes/base/jquery-ui.css">
+<input type="text" id="input_q" style="width:600px" />
+<button onclick="addQa();">新增</button><button onclick="saveQa();">修改項目</button><button
+    onclick="removeQa();">移除項目</button><button onclick="writeAllQa();">完成儲存</button>
+<div style="width:800px;"><textarea id='input_a'></textarea></div>
+<hr />
+<input id="curl" type="text" style="display: none;" /><br />
+<div>
+    <ul id='qaData1'></ul>
+</div>
+<style>
+    #qaData1 li {
+        vertical-align: top;
+    }
+
+    #qaData1 tr {
+        border: inset 2px
+    }
+
+    #qaData1 td {
+        vertical-align: top;
+        border: inset 1px
+    }
+</style>
+<script>
+
+    var editor, targetItem, faqContent;
+    qa = [];
+    $(function () {
+        editor = $("#input_a").richText();
+        axios.get(contentApiUrl + '/frequently_asked_questions').then(({ data }) => {
+            parseMd(data[0]['content']);
+            //alert(contentMatters.join(''));
+            faqContent = $.parseHTML(contentMatters.join(''));
+            //處理圖片
+            if ($("[id='myAccordion']", faqContent).length > 0) {
+                fnodes = $("[id='myAccordion']", faqContent)[0].childNodes;
+                for (i = 0; i < fnodes.length; i++) {
+                    if (fnodes[i].nodeName == "SECTION") {
+                        //qa.push({ q: fnodes[i].childNodes[1].innerText.trim(), a: fnodes[i].childNodes[3].innerHTML.trim() });
+                        k = 0;
+                        for (j = 0; j < fnodes[i].childNodes.length; j++) {
+                            if (fnodes[i].childNodes[j].nodeName == "DIV") {
+                                if (k == 0) {
+                                    qa.push({ q: fnodes[i].childNodes[j].innerText.trim() });
+                                    k++;
+                                }
+                                else {
+                                    qa[qa.length - 1].a = fnodes[i].childNodes[j].innerHTML.trim();
+                                }
+                                /* if (fnodes[i].childNodes[j].className == "question") {
+                                    //alert(fnodes[i].childNodes[j].childNodes[1].nodeValue.trim());
+                                    qa.push({ q: fnodes[i].childNodes[j].childNodes[1].nodeValue.trim() });
+                                } */
+                            }
+                        }
+                    }
+                }
+                //alert(qa.length);
+                for (i = 0; i < qa.length; i++) {
+                    //$('#qaData').append("<tr><td>" + qa[i].q + "</td><td>" + qa[i].a + "</td></tr>");
+                    $('#qaData1').append("<li data-si='" + i + "''><table><tr class='qa_q'><td>" + qa[i].q + "</td></tr><tr class='qa_a'><td>" + qa[i].a + "</td></tr></table></li>");
+
+                }
+                $('.qa_a').hide();
+                $('.qa_q').dblclick(setTarget);
+                /* $('.qa_q').dblclick(function () {
+                    //alert($(this).next()[0].childNodes[0].innerHTML);
+                    //$('#editorjs')[0].innerHTML = "";
+                    //$('.qa_a').hide();
+                    //$(this).next().toggle();
+                    $('.qa_q').css("background-color", "white");
+                    $(this).css("background-color", "lightpink");
+                    targetItem = $(this);
+                    $("#input_q").val($(this).children().html());
+                    $("#input_a").val($(this).next().children().html()).trigger('change');
+                    $('html, body').animate({ scrollTop: '0px' }, 300);
+                    //$("#editor")[0].outerHTML = "<textarea id='editor'></textarea>";
+                    //$("#editor").val($(this).next()[0].childNodes[0].innerHTML);
+                    //editor = $("#editor").richText();
+                    //$(this).next().append("<div id='editorjs'></div>");
+                }); */
+
+                //editor = $("textarea").richText();
+                fromIdx = 0;
+                $('#qaData1').sortable(/* {
+                    start: function (event, ui) {
+                        fromIdx = ui.item.index();
+                    },
+                    stop: function (event, ui) {
+                        //toIndex = ui.item.attr("data-si");
+                        var element = qa[fromIdx];
+                        qa.splice(fromIdx, 1);
+                        qa.splice(ui.item.index(), 0, element);
+                        //alert(fromIdx + ',' + ui.item.index());
+                    },
+                } */);
+            }
+        });
+
+    });
+
+    function setTarget() {
+        $('.qa_q').css("background-color", "white");
+        $(this).css("background-color", "lightpink");
+        targetItem = $(this);
+        $("#input_q").val($(this).children().html());
+        $("#input_a").val($(this).next().children().html()).trigger('change');
+        $('html, body').animate({ scrollTop: '0px' }, 300);
+    }
+
+    function addQa() {
+        $('#qaData1').append("<li data-si='" + i + "''><table><tr class='qa_q'><td>" + $("#input_q").val() + "</td></tr><tr class='qa_a'><td>" + $("#input_a").val() + "</td></tr></table></li>");
+        $('.qa_a').hide();
+        $('.qa_q').dblclick(setTarget);
+    };
+
+    function saveQa() {
+        targetItem.children().html($("#input_q").val());
+        targetItem.next().children().html($("#input_a").val());
+    };
+
+    function removeQa() {
+        targetItem.parent().parent().parent().remove()
+    };
+
+    function writeAllQa() {
+        qa = [];
+        for (i = 0; i < $('#qaData1 .qa_q').length; i++) {
+            qa.push({ q: $('#qaData1 .qa_q')[i].childNodes[0].innerHTML, a: $('#qaData1 .qa_a')[i].childNodes[0].innerHTML });
+        }
+        newFaqItemContent = "";
+        for (i = 0; i < qa.length; i++) {
+            newFaqItemContent += "<section><div class='question'><i class='fa fa-angle-down'></i>" + qa[i].q + "</div><div>" + qa[i].a + "</div></section>";
+        }
+        $("[id='myAccordion']", faqContent).html(newFaqItemContent);
+
+        mdContent = frontMatters.join('\n') + '\n' + faqContent[0].outerHTML
+        //alert(mdContent);
+        postData = {
+            content: mdContent,
+            url: $('#curl').val()
+        };
+        axios.post(contentApiUrl + $('#curl').val(), json = postData).then(({ data }) => {
+            alert('作品資料已儲存');
+        });
+        //alert(faqContent.outerHTML);
+    }
+
+</script>
+{% endblock main %}

+ 1 - 1
backstage/templates/layout.html

@@ -78,7 +78,7 @@
               <span class="menu-collapsed">門市據點</span>
             </div>
           </a>
-          <a href="#" class="bg-dark list-group-item list-group-item-action">
+          <a href="{{ url_for('blogs.edit_faq') }}" class="bg-dark list-group-item list-group-item-action">
             <div class="d-flex w-100 justify-content-start align-items-center">
               <i class="fas fa-question-circle mr-3"></i>
               <span class="menu-collapsed">常見問題</span>

Alguns ficheiros não foram mostrados porque muitos ficheiros mudaram neste diff