1 // Table Operations Plugin for HTMLArea-3.0
2 // Implementation by Mihai Bazon. Sponsored by http://www.bloki.com
4 // htmlArea v3.0 - Copyright (c) 2002 interactivetools.com, inc.
5 // This notice MUST stay intact for use (see license.txt).
7 // A free WYSIWYG editor replacement for <textarea> fields.
8 // For full source code and docs, visit http://www.interactivetools.com/
10 // Version 3.0 developed by Mihai Bazon for InteractiveTools.
11 // http://dynarch.com/mishoo
15 // Object that will encapsulate all the table operations provided by
16 // HTMLArea-3.0 (except "insert table" which is included in the main file)
17 function TableOperations(editor) {
20 var cfg = editor.config;
21 var tt = TableOperations.I18N;
22 var bl = TableOperations.btnList;
25 // register the toolbar buttons provided by this plugin
26 var toolbar = ["linebreak"];
30 toolbar.push("separator");
32 var id = "TO-" + btn[0];
33 cfg.registerButton(id, tt[id], editor.imgURL(btn[0] + ".gif", "TableOperations"), false,
34 function(editor, id) {
35 // dispatch button press event
36 self.buttonPress(editor, id);
42 // add a new line in the toolbar
43 cfg.toolbar.push(toolbar);
46 TableOperations._pluginInfo = {
47 name : "TableOperations",
49 developer : "Mihai Bazon",
50 developer_url : "http://dynarch.com/mishoo/",
51 c_owner : "Mihai Bazon",
52 sponsor : "Zapatec Inc.",
53 sponsor_url : "http://www.bloki.com",
57 /************************
59 ************************/
61 // retrieves the closest element having the specified tagName in the list of
62 // ancestors of the current selection/caret.
63 TableOperations.prototype.getClosest = function(tagName) {
64 var editor = this.editor;
65 var ancestors = editor.getAllAncestors();
67 tagName = ("" + tagName).toLowerCase();
68 for (var i in ancestors) {
69 var el = ancestors[i];
70 if (el.tagName.toLowerCase() == tagName) {
78 // this function requires the file PopupDiv/PopupWin to be loaded from browser
79 TableOperations.prototype.dialogTableProperties = function() {
80 var i18n = TableOperations.I18N;
81 // retrieve existing values
82 var table = this.getClosest("table");
83 // this.editor.selectNodeContents(table);
84 // this.editor.updateToolbar();
86 var dialog = new PopupWin(this.editor, i18n["Table Properties"], function(dialog, params) {
87 TableOperations.processStyle(params, table);
88 for (var i in params) {
93 // contains non white-space characters
94 var caption = table.getElementsByTagName("caption")[0];
96 caption = dialog.editor._doc.createElement("caption");
97 table.insertBefore(caption, table.firstChild);
99 caption.innerHTML = val;
101 // search for caption and delete it if found
102 var caption = table.getElementsByTagName("caption")[0];
104 caption.parentNode.removeChild(caption);
112 table.style.width = ("" + val) + params.f_unit;
118 table.cellSpacing = val;
121 table.cellPadding = val;
134 // various workarounds to refresh the table display (Gecko,
135 // what's going on?! do not disappoint me!)
136 dialog.editor.forceRedraw();
137 dialog.editor.focusEditor();
138 dialog.editor.updateToolbar();
139 var save_collapse = table.style.borderCollapse;
140 table.style.borderCollapse = "collapse";
141 table.style.borderCollapse = "separate";
142 table.style.borderCollapse = save_collapse;
145 // this function gets called when the dialog needs to be initialized
149 var capel = table.getElementsByTagName("caption")[0];
151 f_caption = capel.innerHTML;
153 var f_summary = table.summary;
154 var f_width = parseInt(table.style.width);
155 isNaN(f_width) && (f_width = "");
156 var f_unit = /%/.test(table.style.width) ? 'percent' : 'pixels';
157 var f_align = table.align;
158 var f_spacing = table.cellSpacing;
159 var f_padding = table.cellPadding;
160 var f_borders = table.border;
161 var f_frames = table.frame;
162 var f_rules = table.rules;
164 function selected(val) {
165 return val ? " selected" : "";
170 dialog.content.innerHTML = " \
172 style='background: url(" + dialog.baseURL + dialog.editor.imgURL("table-prop.gif", "TableOperations") + ") #fff 98% 50% no-repeat'>" + i18n["Table Properties"] + "\
174 <table style='width:100%'> \
177 <fieldset><legend>" + i18n["Description"] + "</legend> \
178 <table style='width:100%'> \
180 <td class='label'>" + i18n["Caption"] + ":</td> \
181 <td class='value'><input type='text' name='f_caption' value='" + f_caption + "'/></td> \
183 <td class='label'>" + i18n["Summary"] + ":</td> \
184 <td class='value'><input type='text' name='f_summary' value='" + f_summary + "'/></td> \
190 <tr><td id='--HA-layout'></td></tr> \
193 <fieldset><legend>" + i18n["Spacing and padding"] + "</legend> \
194 <table style='width:100%'> \
196 // <td class='label'>" + i18n["Width"] + ":</td> \
197 // <td><input type='text' name='f_width' value='" + f_width + "' size='5' /> \
198 // <select name='f_unit'> \
199 // <option value='%'" + selected(f_unit == "percent") + ">" + i18n["percent"] + "</option> \
200 // <option value='px'" + selected(f_unit == "pixels") + ">" + i18n["pixels"] + "</option> \
201 // </select> " + i18n["Align"] + ": \
202 // <select name='f_align'> \
203 // <option value='left'" + selected(f_align == "left") + ">" + i18n["Left"] + "</option> \
204 // <option value='center'" + selected(f_align == "center") + ">" + i18n["Center"] + "</option> \
205 // <option value='right'" + selected(f_align == "right") + ">" + i18n["Right"] + "</option> \
210 <td class='label'>" + i18n["Spacing"] + ":</td> \
211 <td><input type='text' name='f_spacing' size='5' value='" + f_spacing + "' /> " + i18n["Padding"] + ":\
212 <input type='text' name='f_padding' size='5' value='" + f_padding + "' /> " + i18n["pixels"] + "\
221 <fieldset><legend>Frame and borders</legend> \
222 <table width='100%'> \
224 <td class='label'>" + i18n["Borders"] + ":</td> \
225 <td><input name='f_borders' type='text' size='5' value='" + f_borders + "' /> " + i18n["pixels"] + "</td> \
228 <td class='label'>" + i18n["Frames"] + ":</td> \
230 <select name='f_frames'> \
231 <option value='void'" + selected(f_frames == "void") + ">" + i18n["No sides"] + "</option> \
232 <option value='above'" + selected(f_frames == "above") + ">" + i18n["The top side only"] + "</option> \
233 <option value='below'" + selected(f_frames == "below") + ">" + i18n["The bottom side only"] + "</option> \
234 <option value='hsides'" + selected(f_frames == "hsides") + ">" + i18n["The top and bottom sides only"] + "</option> \
235 <option value='vsides'" + selected(f_frames == "vsides") + ">" + i18n["The right and left sides only"] + "</option> \
236 <option value='lhs'" + selected(f_frames == "lhs") + ">" + i18n["The left-hand side only"] + "</option> \
237 <option value='rhs'" + selected(f_frames == "rhs") + ">" + i18n["The right-hand side only"] + "</option> \
238 <option value='box'" + selected(f_frames == "box") + ">" + i18n["All four sides"] + "</option> \
243 <td class='label'>" + i18n["Rules"] + ":</td> \
245 <select name='f_rules'> \
246 <option value='none'" + selected(f_rules == "none") + ">" + i18n["No rules"] + "</option> \
247 <option value='rows'" + selected(f_rules == "rows") + ">" + i18n["Rules will appear between rows only"] + "</option> \
248 <option value='cols'" + selected(f_rules == "cols") + ">" + i18n["Rules will appear between columns only"] + "</option> \
249 <option value='all'" + selected(f_rules == "all") + ">" + i18n["Rules will appear between all rows and columns"] + "</option> \
258 <td id='--HA-style'></td> \
262 var st_prop = TableOperations.createStyleFieldset(dialog.doc, dialog.editor, table);
263 var p = dialog.doc.getElementById("--HA-style");
264 p.appendChild(st_prop);
265 var st_layout = TableOperations.createStyleLayoutFieldset(dialog.doc, dialog.editor, table);
266 p = dialog.doc.getElementById("--HA-layout");
267 p.appendChild(st_layout);
269 dialog.addButtons("ok", "cancel");
270 dialog.showAtElement(dialog.editor._iframe, "c");
271 dialog.content.style.width = "400px";
273 dialog.content.style.height = dialog.content.clientHeight + 60 + 'px'; //moodlefix
278 // this function requires the file PopupDiv/PopupWin to be loaded from browser
279 TableOperations.prototype.dialogRowCellProperties = function(cell) {
280 var i18n = TableOperations.I18N;
281 // retrieve existing values
282 var element = this.getClosest(cell ? "td" : "tr");
283 var table = this.getClosest("table");
284 // this.editor.selectNodeContents(element);
285 // this.editor.updateToolbar();
287 var dialog = new PopupWin(this.editor, i18n[cell ? "Cell Properties" : "Row Properties"], function(dialog, params) {
288 TableOperations.processStyle(params, element);
289 for (var i in params) {
299 element.vAlign = val;
303 // various workarounds to refresh the table display (Gecko,
304 // what's going on?! do not disappoint me!)
305 dialog.editor.forceRedraw();
306 dialog.editor.focusEditor();
307 dialog.editor.updateToolbar();
308 var save_collapse = table.style.borderCollapse;
309 table.style.borderCollapse = "collapse";
310 table.style.borderCollapse = "separate";
311 table.style.borderCollapse = save_collapse;
314 // this function gets called when the dialog needs to be initialized
317 var f_align = element.align;
318 var f_valign = element.vAlign;
319 var f_char = element.ch;
321 function selected(val) {
322 return val ? " selected" : "";
326 dialog.content.style.width = "400px";
327 dialog.content.innerHTML = " \
329 style='background: url(" + dialog.baseURL + dialog.editor.imgURL(cell ? "cell-prop.gif" : "row-prop.gif", "TableOperations") + ") #fff 98% 50% no-repeat'>" + i18n[cell ? "Cell Properties" : "Row Properties"] + "</div> \
330 <table style='width:100%'> \
332 <td id='--HA-layout'> \
333 "+// <fieldset><legend>" + i18n["Layout"] + "</legend> \
334 // <table style='width:100%'> \
336 // <td class='label'>" + i18n["Align"] + ":</td> \
338 // <select name='f_align'> \
339 // <option value='left'" + selected(f_align == "left") + ">" + i18n["Left"] + "</option> \
340 // <option value='center'" + selected(f_align == "center") + ">" + i18n["Center"] + "</option> \
341 // <option value='right'" + selected(f_align == "right") + ">" + i18n["Right"] + "</option> \
342 // <option value='char'" + selected(f_align == "char") + ">" + i18n["Char"] + "</option> \
344 // " + i18n["Char"] + ": \
345 // <input type='text' style='font-family: monospace; text-align: center' name='f_char' size='1' value='" + f_char + "' /> \
348 // <td class='label'>" + i18n["Vertical align"] + ":</td> \
350 // <select name='f_valign'> \
351 // <option value='top'" + selected(f_valign == "top") + ">" + i18n["Top"] + "</option> \
352 // <option value='middle'" + selected(f_valign == "middle") + ">" + i18n["Middle"] + "</option> \
353 // <option value='bottom'" + selected(f_valign == "bottom") + ">" + i18n["Bottom"] + "</option> \
354 // <option value='baseline'" + selected(f_valign == "baseline") + ">" + i18n["Baseline"] + "</option> \
363 <td id='--HA-style'></td> \
367 var st_prop = TableOperations.createStyleFieldset(dialog.doc, dialog.editor, element);
368 var p = dialog.doc.getElementById("--HA-style");
369 p.appendChild(st_prop);
370 var st_layout = TableOperations.createStyleLayoutFieldset(dialog.doc, dialog.editor, element);
371 p = dialog.doc.getElementById("--HA-layout");
372 p.appendChild(st_layout);
374 dialog.addButtons("ok", "cancel");
375 dialog.showAtElement(dialog.editor._iframe, "c");
377 dialog.content.style.height = dialog.content.clientHeight + 60 + 'px'; //moodlefix
382 // this function gets called when some button from the TableOperations toolbar
384 TableOperations.prototype.buttonPress = function(editor, button_id) {
385 this.editor = editor;
386 var mozbr = HTMLArea.is_gecko ? "<br />" : "";
387 var i18n = TableOperations.I18N;
389 // helper function that clears the content in a table row
390 function clearRow(tr) {
391 var tds = tr.getElementsByTagName("td");
392 for (var i = tds.length; --i >= 0;) {
395 td.innerHTML = mozbr;
399 function splitRow(td) {
400 var n = parseInt("" + td.rowSpan);
401 var nc = parseInt("" + td.colSpan);
404 var itr = tr.rowIndex;
405 var trs = tr.parentNode.rows;
406 var index = td.cellIndex;
409 var otd = editor._doc.createElement("td");
410 otd.colSpan = td.colSpan;
411 otd.innerHTML = mozbr;
412 tr.insertBefore(otd, tr.cells[index]);
414 editor.forceRedraw();
415 editor.updateToolbar();
418 function splitCol(td) {
419 var nc = parseInt("" + td.colSpan);
422 var ref = td.nextSibling;
424 var otd = editor._doc.createElement("td");
425 otd.rowSpan = td.rowSpan;
426 otd.innerHTML = mozbr;
427 tr.insertBefore(otd, ref);
429 editor.forceRedraw();
430 editor.updateToolbar();
433 function splitCell(td) {
434 var nc = parseInt("" + td.colSpan);
436 var items = td.parentNode.cells;
437 var index = td.cellIndex;
439 splitRow(items[index++]);
443 function selectNextNode(el) {
444 var node = el.nextSibling;
445 while (node && node.nodeType != 1) {
446 node = node.nextSibling;
449 node = el.previousSibling;
450 while (node && node.nodeType != 1) {
451 node = node.previousSibling;
455 node = el.parentNode;
457 editor.selectNodeContents(node);
463 case "TO-row-insert-above":
464 case "TO-row-insert-under":
465 var tr = this.getClosest("tr");
469 var otr = tr.cloneNode(true);
471 tr.parentNode.insertBefore(otr, /under/.test(button_id) ? tr.nextSibling : tr);
472 editor.forceRedraw();
473 editor.focusEditor();
475 case "TO-row-delete":
476 var tr = this.getClosest("tr");
480 var par = tr.parentNode;
481 if (par.rows.length == 1) {
482 alert(i18n["not-del-last-row"]);
485 // set the caret first to a position that doesn't
489 editor.forceRedraw();
490 editor.focusEditor();
491 editor.updateToolbar();
494 var td = this.getClosest("td");
503 case "TO-col-insert-before":
504 case "TO-col-insert-after":
505 var td = this.getClosest("td");
509 var rows = td.parentNode.parentNode.rows;
510 var index = td.cellIndex;
511 for (var i = rows.length; --i >= 0;) {
513 var ref = tr.cells[index + (/after/.test(button_id) ? 1 : 0)];
514 var otd = editor._doc.createElement("td");
515 otd.innerHTML = mozbr;
516 tr.insertBefore(otd, ref);
518 editor.focusEditor();
521 var td = this.getClosest("td");
527 case "TO-col-delete":
528 var td = this.getClosest("td");
532 var index = td.cellIndex;
533 if (td.parentNode.cells.length == 1) {
534 alert(i18n["not-del-last-col"]);
537 // set the caret first to a position that doesn't disappear
539 var rows = td.parentNode.parentNode.rows;
540 for (var i = rows.length; --i >= 0;) {
542 tr.removeChild(tr.cells[index]);
544 editor.forceRedraw();
545 editor.focusEditor();
546 editor.updateToolbar();
551 case "TO-cell-split":
552 var td = this.getClosest("td");
558 case "TO-cell-insert-before":
559 case "TO-cell-insert-after":
560 var td = this.getClosest("td");
564 var tr = td.parentNode;
565 var otd = editor._doc.createElement("td");
566 otd.innerHTML = mozbr;
567 tr.insertBefore(otd, /after/.test(button_id) ? td.nextSibling : td);
568 editor.forceRedraw();
569 editor.focusEditor();
571 case "TO-cell-delete":
572 var td = this.getClosest("td");
576 if (td.parentNode.cells.length == 1) {
577 alert(i18n["not-del-last-cell"]);
580 // set the caret first to a position that doesn't disappear
582 td.parentNode.removeChild(td);
583 editor.forceRedraw();
584 editor.updateToolbar();
586 case "TO-cell-merge":
587 // !! FIXME: Mozilla specific !!
588 var sel = editor._getSelection();
593 if (!HTMLArea.is_ie) {
595 while (range = sel.getRangeAt(i++)) {
596 var td = range.startContainer.childNodes[range.startOffset];
597 if (td.parentNode != row) {
599 (cells) && rows.push(cells);
604 } catch(e) {/* finished walking through selection */}
607 // Internet Explorer "browser"
608 var td = this.getClosest("td");
610 alert(i18n["Please click into some cell"]);
613 var tr = td.parentElement;
614 var no_cols = prompt(i18n["How many columns would you like to merge?"], 2);
619 var no_rows = prompt(i18n["How many rows would you like to merge?"], 2);
624 var cell_index = td.cellIndex;
625 while (no_rows-- > 0) {
626 td = tr.cells[cell_index];
628 for (var i = 1; i < no_cols; ++i) {
643 for (i = 0; i < rows.length; ++i) {
644 // i && (HTML += "<br />");
646 for (var j = 0; j < cells.length; ++j) {
647 // j && (HTML += " ");
649 HTML += cell.innerHTML;
650 (i || j) && (cell.parentNode.removeChild(cell));
655 td.rowSpan = rows.length;
656 td.colSpan = rows[0].length;
657 editor.selectNodeContents(td);
658 editor.forceRedraw();
659 editor.focusEditor();
664 case "TO-table-prop":
665 this.dialogTableProperties();
669 this.dialogRowCellProperties(false);
673 this.dialogRowCellProperties(true);
677 alert("Button [" + button_id + "] not yet implemented");
681 // the list of buttons added by this plugin
682 TableOperations.btnList = [
683 // table properties button
684 ["table-prop", "table"],
689 ["row-insert-above", "tr"],
690 ["row-insert-under", "tr"],
691 ["row-delete", "tr"],
692 ["row-split", "td[rowSpan!=1]"],
696 ["col-insert-before", "td"],
697 ["col-insert-after", "td"],
698 ["col-delete", "td"],
699 ["col-split", "td[colSpan!=1]"],
704 ["cell-insert-before", "td"],
705 ["cell-insert-after", "td"],
706 ["cell-delete", "td"],
707 ["cell-merge", "tr"],
708 ["cell-split", "td[colSpan!=1,rowSpan!=1]"]
713 //// GENERIC CODE [style of any element; this should be moved into a separate
714 //// file as it'll be very useful]
715 //// BEGIN GENERIC CODE -----------------------------------------------------
717 TableOperations.getLength = function(value) {
718 var len = parseInt(value);
725 // Applies the style found in "params" to the given element.
726 TableOperations.processStyle = function(params, element) {
727 var style = element.style;
728 for (var i in params) {
731 case "f_st_backgroundColor":
732 style.backgroundColor = val;
737 case "f_st_backgroundImage":
738 if (/\S/.test(val)) {
739 style.backgroundImage = "url(" + val + ")";
741 style.backgroundImage = "none";
744 case "f_st_borderWidth":
745 style.borderWidth = val;
747 case "f_st_borderStyle":
748 style.borderStyle = val;
750 case "f_st_borderColor":
751 style.borderColor = val;
753 case "f_st_borderCollapse":
754 style.borderCollapse = val ? "collapse" : "";
757 if (/\S/.test(val)) {
758 style.width = val + params["f_st_widthUnit"];
764 if (/\S/.test(val)) {
765 style.height = val + params["f_st_heightUnit"];
770 case "f_st_textAlign":
772 var ch = params["f_st_textAlignChar"];
776 style.textAlign = '"' + ch + '"';
778 style.textAlign = val;
781 case "f_st_verticalAlign":
782 style.verticalAlign = val;
785 style.cssFloat = val;
787 // case "f_st_margin":
788 // style.margin = val + "px";
790 // case "f_st_padding":
791 // style.padding = val + "px";
797 // Returns an HTML element for a widget that allows color selection. That is,
798 // a button that contains the given color, if any, and when pressed will popup
799 // the sooner-or-later-to-be-rewritten select_color.html dialog allowing user
800 // to select some color. If a color is selected, an input field with the name
801 // "f_st_"+name will be updated with the color value in #123456 format.
802 TableOperations.createColorButton = function(doc, editor, color, name) {
805 } else if (!/#/.test(color)) {
806 color = HTMLArea._colorToRgb(color);
809 var df = doc.createElement("span");
810 var field = doc.createElement("input");
811 field.type = "hidden";
812 df.appendChild(field);
813 field.name = "f_st_" + name;
815 var button = doc.createElement("span");
816 button.className = "buttonColor";
817 df.appendChild(button);
818 var span = doc.createElement("span");
819 span.className = "chooser";
820 // span.innerHTML = " ";
821 span.style.backgroundColor = color;
822 button.appendChild(span);
823 button.onmouseover = function() { if (!this.disabled) { this.className += " buttonColor-hilite"; }};
824 button.onmouseout = function() { if (!this.disabled) { this.className = "buttonColor"; }};
825 span.onclick = function() {
826 if (this.parentNode.disabled) {
829 editor._popupDialog("select_color.php", function(color) {
831 span.style.backgroundColor = "#" + color;
832 field.value = "#" + color;
836 var span2 = doc.createElement("span");
837 span2.innerHTML = "×";
838 span2.className = "nocolor";
839 span2.title = TableOperations.I18N["Unset color"];
840 button.appendChild(span2);
841 span2.onmouseover = function() { if (!this.parentNode.disabled) { this.className += " nocolor-hilite"; }};
842 span2.onmouseout = function() { if (!this.parentNode.disabled) { this.className = "nocolor"; }};
843 span2.onclick = function() {
844 span.style.backgroundColor = "";
850 TableOperations.createStyleLayoutFieldset = function(doc, editor, el) {
851 var i18n = TableOperations.I18N;
852 var fieldset = doc.createElement("fieldset");
853 var legend = doc.createElement("legend");
854 fieldset.appendChild(legend);
855 legend.innerHTML = i18n["Layout"];
856 var table = doc.createElement("table");
857 fieldset.appendChild(table);
858 table.style.width = "100%";
859 var tbody = doc.createElement("tbody");
860 table.appendChild(tbody);
862 var tagname = el.tagName.toLowerCase();
863 var tr, td, input, select, option, options, i;
865 if (tagname != "td" && tagname != "tr" && tagname != "th") {
866 tr = doc.createElement("tr");
867 tbody.appendChild(tr);
868 td = doc.createElement("td");
869 td.className = "label";
871 td.innerHTML = i18n["Float"] + ":";
872 td = doc.createElement("td");
874 select = doc.createElement("select");
875 td.appendChild(select);
876 select.name = "f_st_float";
877 options = ["None", "Left", "Right"];
879 var Val = options[i];
880 var val = options[i].toLowerCase();
881 option = doc.createElement("option");
882 option.innerHTML = i18n[Val];
884 option.selected = (("" + el.style.cssFloat).toLowerCase() == val);
885 select.appendChild(option);
889 tr = doc.createElement("tr");
890 tbody.appendChild(tr);
891 td = doc.createElement("td");
892 td.className = "label";
894 td.innerHTML = i18n["Width"] + ":";
895 td = doc.createElement("td");
897 input = doc.createElement("input");
899 input.value = TableOperations.getLength(el.style.width);
901 input.name = "f_st_width";
902 input.style.marginRight = "0.5em";
903 td.appendChild(input);
904 select = doc.createElement("select");
905 select.name = "f_st_widthUnit";
906 option = doc.createElement("option");
907 option.innerHTML = i18n["percent"];
909 option.selected = /%/.test(el.style.width);
910 select.appendChild(option);
911 option = doc.createElement("option");
912 option.innerHTML = i18n["pixels"];
914 option.selected = /px/.test(el.style.width);
915 select.appendChild(option);
916 td.appendChild(select);
918 select.style.marginRight = "0.5em";
919 td.appendChild(doc.createTextNode(i18n["Text align"] + ":"));
920 select = doc.createElement("select");
921 select.style.marginLeft = select.style.marginRight = "0.5em";
922 td.appendChild(select);
923 select.name = "f_st_textAlign";
924 options = ["Left", "Center", "Right", "Justify"];
925 if (tagname == "td") {
926 options.push("Char");
928 input = doc.createElement("input");
929 input.name = "f_st_textAlignChar";
931 input.style.fontFamily = "monospace";
932 td.appendChild(input);
934 var Val = options[i];
935 var val = Val.toLowerCase();
936 option = doc.createElement("option");
938 option.innerHTML = i18n[Val];
939 option.selected = (el.style.textAlign.toLowerCase() == val);
940 select.appendChild(option);
942 function setCharVisibility(value) {
943 input.style.visibility = value ? "visible" : "hidden";
949 select.onchange = function() { setCharVisibility(this.value == "char"); };
950 setCharVisibility(select.value == "char");
952 tr = doc.createElement("tr");
953 tbody.appendChild(tr);
954 td = doc.createElement("td");
955 td.className = "label";
957 td.innerHTML = i18n["Height"] + ":";
958 td = doc.createElement("td");
960 input = doc.createElement("input");
962 input.value = TableOperations.getLength(el.style.height);
964 input.name = "f_st_height";
965 input.style.marginRight = "0.5em";
966 td.appendChild(input);
967 select = doc.createElement("select");
968 select.name = "f_st_heightUnit";
969 option = doc.createElement("option");
970 option.innerHTML = i18n["percent"];
972 option.selected = /%/.test(el.style.height);
973 select.appendChild(option);
974 option = doc.createElement("option");
975 option.innerHTML = i18n["pixels"];
977 option.selected = /px/.test(el.style.height);
978 select.appendChild(option);
979 td.appendChild(select);
981 select.style.marginRight = "0.5em";
982 td.appendChild(doc.createTextNode(i18n["Vertical align"] + ":"));
983 select = doc.createElement("select");
984 select.name = "f_st_verticalAlign";
985 select.style.marginLeft = "0.5em";
986 td.appendChild(select);
987 options = ["Top", "Middle", "Bottom", "Baseline"];
989 var Val = options[i];
990 var val = Val.toLowerCase();
991 option = doc.createElement("option");
993 option.innerHTML = i18n[Val];
994 option.selected = (el.style.verticalAlign.toLowerCase() == val);
995 select.appendChild(option);
1001 // Returns an HTML element containing the style attributes for the given
1002 // element. This can be easily embedded into any dialog; the functionality is
1004 TableOperations.createStyleFieldset = function(doc, editor, el) {
1005 var i18n = TableOperations.I18N;
1006 var fieldset = doc.createElement("fieldset");
1007 var legend = doc.createElement("legend");
1008 fieldset.appendChild(legend);
1009 legend.innerHTML = i18n["CSS Style"];
1010 var table = doc.createElement("table");
1011 fieldset.appendChild(table);
1012 table.style.width = "100%";
1013 var tbody = doc.createElement("tbody");
1014 table.appendChild(tbody);
1016 var tr, td, input, select, option, options, i;
1018 tr = doc.createElement("tr");
1019 tbody.appendChild(tr);
1020 td = doc.createElement("td");
1022 td.className = "label";
1023 td.innerHTML = i18n["Background"] + ":";
1024 td = doc.createElement("td");
1026 var df = TableOperations.createColorButton(doc, editor, el.style.backgroundColor, "backgroundColor");
1027 df.firstChild.nextSibling.style.marginRight = "0.5em";
1029 td.appendChild(doc.createTextNode(i18n["Image URL"] + ": "));
1030 input = doc.createElement("input");
1031 input.type = "text";
1032 input.name = "f_st_backgroundImage";
1033 if (el.style.backgroundImage.match(/url\(\s*(.*?)\s*\)/)) {
1034 input.value = RegExp.$1;
1036 // input.style.width = "100%";
1037 td.appendChild(input);
1039 tr = doc.createElement("tr");
1040 tbody.appendChild(tr);
1041 td = doc.createElement("td");
1043 td.className = "label";
1044 td.innerHTML = i18n["FG Color"] + ":";
1045 td = doc.createElement("td");
1047 td.appendChild(TableOperations.createColorButton(doc, editor, el.style.color, "color"));
1049 // for better alignment we include an invisible field.
1050 input = doc.createElement("input");
1051 input.style.visibility = "hidden";
1052 input.type = "text";
1053 td.appendChild(input);
1055 tr = doc.createElement("tr");
1056 tbody.appendChild(tr);
1057 td = doc.createElement("td");
1059 td.className = "label";
1060 td.innerHTML = i18n["Border"] + ":";
1061 td = doc.createElement("td");
1064 var colorButton = TableOperations.createColorButton(doc, editor, el.style.borderColor, "borderColor");
1065 var btn = colorButton.firstChild.nextSibling;
1066 td.appendChild(colorButton);
1067 // borderFields.push(btn);
1068 btn.style.marginRight = "0.5em";
1070 select = doc.createElement("select");
1071 var borderFields = [];
1072 td.appendChild(select);
1073 select.name = "f_st_borderStyle";
1074 options = ["none", "dotted", "dashed", "solid", "double", "groove", "ridge", "inset", "outset"];
1075 var currentBorderStyle = el.style.borderStyle;
1076 // Gecko reports "solid solid solid solid" for "border-style: solid".
1077 // That is, "top right bottom left" -- we only consider the first
1079 (currentBorderStyle.match(/([^\s]*)\s/)) && (currentBorderStyle = RegExp.$1);
1080 for (i in options) {
1081 var val = options[i];
1082 option = doc.createElement("option");
1084 option.innerHTML = val;
1085 (val == currentBorderStyle) && (option.selected = true);
1086 select.appendChild(option);
1088 select.style.marginRight = "0.5em";
1089 function setBorderFieldsStatus(value) {
1090 for (i in borderFields) {
1091 var el = borderFields[i];
1092 el.style.visibility = value ? "hidden" : "visible";
1093 if (!value && (el.tagName.toLowerCase() == "input")) {
1099 select.onchange = function() { setBorderFieldsStatus(this.value == "none"); };
1101 input = doc.createElement("input");
1102 borderFields.push(input);
1103 input.type = "text";
1104 input.name = "f_st_borderWidth";
1105 input.value = TableOperations.getLength(el.style.borderWidth);
1107 td.appendChild(input);
1108 input.style.marginRight = "0.5em";
1109 var span = doc.createElement("span");
1110 span.innerHTML = i18n["pixels"];
1111 td.appendChild(span);
1112 borderFields.push(span);
1114 setBorderFieldsStatus(select.value == "none");
1116 if (el.tagName.toLowerCase() == "table") {
1117 // the border-collapse style is only for tables
1118 tr = doc.createElement("tr");
1119 tbody.appendChild(tr);
1120 td = doc.createElement("td");
1121 td.className = "label";
1123 input = doc.createElement("input");
1124 input.type = "checkbox";
1125 input.name = "f_st_borderCollapse";
1126 input.id = "f_st_borderCollapse";
1127 var val = (/collapse/i.test(el.style.borderCollapse));
1128 input.checked = val ? 1 : 0;
1129 td.appendChild(input);
1131 td = doc.createElement("td");
1133 var label = doc.createElement("label");
1134 label.htmlFor = "f_st_borderCollapse";
1135 label.innerHTML = i18n["Collapsed borders"];
1136 td.appendChild(label);
1139 // tr = doc.createElement("tr");
1140 // tbody.appendChild(tr);
1141 // td = doc.createElement("td");
1142 // td.className = "label";
1143 // tr.appendChild(td);
1144 // td.innerHTML = i18n["Margin"] + ":";
1145 // td = doc.createElement("td");
1146 // tr.appendChild(td);
1147 // input = doc.createElement("input");
1148 // input.type = "text";
1149 // input.size = "5";
1150 // input.name = "f_st_margin";
1151 // td.appendChild(input);
1152 // input.style.marginRight = "0.5em";
1153 // td.appendChild(doc.createTextNode(i18n["Padding"] + ":"));
1155 // input = doc.createElement("input");
1156 // input.type = "text";
1157 // input.size = "5";
1158 // input.name = "f_st_padding";
1159 // td.appendChild(input);
1160 // input.style.marginLeft = "0.5em";
1161 // input.style.marginRight = "0.5em";
1162 // td.appendChild(doc.createTextNode(i18n["pixels"]));
1167 //// END GENERIC CODE -------------------------------------------------------