root / static / js / tinymce / plugins / table / editor_plugin_src.js @ 6ecbf4ec
History | View | Annotate | Download (28.6 kB)
1 |
/**
|
---|---|
2 |
* editor_plugin_src.js
|
3 |
*
|
4 |
* Copyright 2009, Moxiecode Systems AB
|
5 |
* Released under LGPL License.
|
6 |
*
|
7 |
* License: http://tinymce.moxiecode.com/license
|
8 |
* Contributing: http://tinymce.moxiecode.com/contributing
|
9 |
*/
|
10 |
|
11 |
(function(tinymce) {
|
12 |
var each = tinymce.each;
|
13 |
|
14 |
/**
|
15 |
* Table Grid class.
|
16 |
*/
|
17 |
function TableGrid(table, dom, selection) { |
18 |
var grid, startPos, endPos, selectedCell;
|
19 |
|
20 |
buildGrid(); |
21 |
selectedCell = dom.getParent(selection.getStart(), 'th,td');
|
22 |
if (selectedCell) {
|
23 |
startPos = getPos(selectedCell); |
24 |
endPos = findEndPos(); |
25 |
selectedCell = getCell(startPos.x, startPos.y); |
26 |
} |
27 |
|
28 |
function cloneNode(node, children) { |
29 |
node = node.cloneNode(children); |
30 |
node.removeAttribute('id');
|
31 |
|
32 |
return node;
|
33 |
} |
34 |
|
35 |
function buildGrid() { |
36 |
var startY = 0; |
37 |
|
38 |
grid = []; |
39 |
|
40 |
each(['thead', 'tbody', 'tfoot'], function(part) { |
41 |
var rows = dom.select(part + ' tr', table); |
42 |
|
43 |
each(rows, function(tr, y) {
|
44 |
y += startY; |
45 |
|
46 |
each(dom.select('td,th', tr), function(td, x) { |
47 |
var x2, y2, rowspan, colspan;
|
48 |
|
49 |
// Skip over existing cells produced by rowspan
|
50 |
if (grid[y]) {
|
51 |
while (grid[y][x])
|
52 |
x++; |
53 |
} |
54 |
|
55 |
// Get col/rowspan from cell
|
56 |
rowspan = getSpanVal(td, 'rowspan');
|
57 |
colspan = getSpanVal(td, 'colspan');
|
58 |
|
59 |
// Fill out rowspan/colspan right and down
|
60 |
for (y2 = y; y2 < y + rowspan; y2++) {
|
61 |
if (!grid[y2])
|
62 |
grid[y2] = []; |
63 |
|
64 |
for (x2 = x; x2 < x + colspan; x2++) {
|
65 |
grid[y2][x2] = { |
66 |
part : part,
|
67 |
real : y2 == y && x2 == x,
|
68 |
elm : td,
|
69 |
rowspan : rowspan,
|
70 |
colspan : colspan
|
71 |
}; |
72 |
} |
73 |
} |
74 |
}); |
75 |
}); |
76 |
|
77 |
startY += rows.length; |
78 |
}); |
79 |
}; |
80 |
|
81 |
function getCell(x, y) { |
82 |
var row;
|
83 |
|
84 |
row = grid[y]; |
85 |
if (row)
|
86 |
return row[x];
|
87 |
}; |
88 |
|
89 |
function getSpanVal(td, name) { |
90 |
return parseInt(td.getAttribute(name) || 1); |
91 |
}; |
92 |
|
93 |
function isCellSelected(cell) { |
94 |
return dom.hasClass(cell.elm, 'mceSelected') || cell == selectedCell; |
95 |
}; |
96 |
|
97 |
function getSelectedRows() { |
98 |
var rows = [];
|
99 |
|
100 |
each(table.rows, function(row) {
|
101 |
each(row.cells, function(cell) {
|
102 |
if (dom.hasClass(cell, 'mceSelected') || cell == selectedCell.elm) { |
103 |
rows.push(row); |
104 |
return false; |
105 |
} |
106 |
}); |
107 |
}); |
108 |
|
109 |
return rows;
|
110 |
}; |
111 |
|
112 |
function deleteTable() { |
113 |
var rng = dom.createRng();
|
114 |
|
115 |
rng.setStartAfter(table); |
116 |
rng.setEndAfter(table); |
117 |
|
118 |
selection.setRng(rng); |
119 |
|
120 |
dom.remove(table); |
121 |
}; |
122 |
|
123 |
function cloneCell(cell) { |
124 |
var formatNode;
|
125 |
|
126 |
// Clone formats
|
127 |
tinymce.walk(cell, function(node) {
|
128 |
var curNode;
|
129 |
|
130 |
if (node.nodeType == 3) { |
131 |
each(dom.getParents(node.parentNode, null, cell).reverse(), function(node) { |
132 |
node = cloneNode(node, false);
|
133 |
|
134 |
if (!formatNode)
|
135 |
formatNode = curNode = node; |
136 |
else if (curNode) |
137 |
curNode.appendChild(node); |
138 |
|
139 |
curNode = node; |
140 |
}); |
141 |
|
142 |
// Add something to the inner node
|
143 |
if (curNode)
|
144 |
curNode.innerHTML = tinymce.isIE ? ' ' : '<br _mce_bogus="1" />'; |
145 |
|
146 |
return false; |
147 |
} |
148 |
}, 'childNodes');
|
149 |
|
150 |
cell = cloneNode(cell, false);
|
151 |
cell.rowSpan = cell.colSpan = 1;
|
152 |
|
153 |
if (formatNode) {
|
154 |
cell.appendChild(formatNode); |
155 |
} else {
|
156 |
if (!tinymce.isIE)
|
157 |
cell.innerHTML = '<br _mce_bogus="1" />';
|
158 |
} |
159 |
|
160 |
return cell;
|
161 |
}; |
162 |
|
163 |
function cleanup() { |
164 |
var rng = dom.createRng();
|
165 |
|
166 |
// Empty rows
|
167 |
each(dom.select('tr', table), function(tr) { |
168 |
if (tr.cells.length == 0) |
169 |
dom.remove(tr); |
170 |
}); |
171 |
|
172 |
// Empty table
|
173 |
if (dom.select('tr', table).length == 0) { |
174 |
rng.setStartAfter(table); |
175 |
rng.setEndAfter(table); |
176 |
selection.setRng(rng); |
177 |
dom.remove(table); |
178 |
return;
|
179 |
} |
180 |
|
181 |
// Empty header/body/footer
|
182 |
each(dom.select('thead,tbody,tfoot', table), function(part) { |
183 |
if (part.rows.length == 0) |
184 |
dom.remove(part); |
185 |
}); |
186 |
|
187 |
// Restore selection to start position if it still exists
|
188 |
buildGrid(); |
189 |
|
190 |
// Restore the selection to the closest table position
|
191 |
row = grid[Math.min(grid.length - 1, startPos.y)];
|
192 |
if (row) {
|
193 |
selection.select(row[Math.min(row.length - 1, startPos.x)].elm, true); |
194 |
selection.collapse(true);
|
195 |
} |
196 |
}; |
197 |
|
198 |
function fillLeftDown(x, y, rows, cols) { |
199 |
var tr, x2, r, c, cell;
|
200 |
|
201 |
tr = grid[y][x].elm.parentNode; |
202 |
for (r = 1; r <= rows; r++) { |
203 |
tr = dom.getNext(tr, 'tr');
|
204 |
|
205 |
if (tr) {
|
206 |
// Loop left to find real cell
|
207 |
for (x2 = x; x2 >= 0; x2--) { |
208 |
cell = grid[y + r][x2].elm; |
209 |
|
210 |
if (cell.parentNode == tr) {
|
211 |
// Append clones after
|
212 |
for (c = 1; c <= cols; c++) |
213 |
dom.insertAfter(cloneCell(cell), cell); |
214 |
|
215 |
break;
|
216 |
} |
217 |
} |
218 |
|
219 |
if (x2 == -1) { |
220 |
// Insert nodes before first cell
|
221 |
for (c = 1; c <= cols; c++) |
222 |
tr.insertBefore(cloneCell(tr.cells[0]), tr.cells[0]); |
223 |
} |
224 |
} |
225 |
} |
226 |
}; |
227 |
|
228 |
function split() { |
229 |
each(grid, function(row, y) {
|
230 |
each(row, function(cell, x) {
|
231 |
var colSpan, rowSpan, newCell, i;
|
232 |
|
233 |
if (isCellSelected(cell)) {
|
234 |
cell = cell.elm; |
235 |
colSpan = getSpanVal(cell, 'colspan');
|
236 |
rowSpan = getSpanVal(cell, 'rowspan');
|
237 |
|
238 |
if (colSpan > 1 || rowSpan > 1) { |
239 |
cell.colSpan = cell.rowSpan = 1;
|
240 |
|
241 |
// Insert cells right
|
242 |
for (i = 0; i < colSpan - 1; i++) |
243 |
dom.insertAfter(cloneCell(cell), cell); |
244 |
|
245 |
fillLeftDown(x, y, rowSpan - 1, colSpan);
|
246 |
} |
247 |
} |
248 |
}); |
249 |
}); |
250 |
}; |
251 |
|
252 |
function merge(cell, cols, rows) { |
253 |
var startX, startY, endX, endY, x, y, startCell, endCell, cell, children;
|
254 |
|
255 |
// Use specified cell and cols/rows
|
256 |
if (cell) {
|
257 |
pos = getPos(cell); |
258 |
startX = pos.x; |
259 |
startY = pos.y; |
260 |
endX = startX + (cols - 1);
|
261 |
endY = startY + (rows - 1);
|
262 |
} else {
|
263 |
// Use selection
|
264 |
startX = startPos.x; |
265 |
startY = startPos.y; |
266 |
endX = endPos.x; |
267 |
endY = endPos.y; |
268 |
} |
269 |
|
270 |
// Find start/end cells
|
271 |
startCell = getCell(startX, startY); |
272 |
endCell = getCell(endX, endY); |
273 |
|
274 |
// Check if the cells exists and if they are of the same part for example tbody = tbody
|
275 |
if (startCell && endCell && startCell.part == endCell.part) {
|
276 |
// Split and rebuild grid
|
277 |
split(); |
278 |
buildGrid(); |
279 |
|
280 |
// Set row/col span to start cell
|
281 |
startCell = getCell(startX, startY).elm; |
282 |
startCell.colSpan = (endX - startX) + 1;
|
283 |
startCell.rowSpan = (endY - startY) + 1;
|
284 |
|
285 |
// Remove other cells and add it's contents to the start cell
|
286 |
for (y = startY; y <= endY; y++) {
|
287 |
for (x = startX; x <= endX; x++) {
|
288 |
cell = grid[y][x].elm; |
289 |
|
290 |
if (cell != startCell) {
|
291 |
// Move children to startCell
|
292 |
children = tinymce.grep(cell.childNodes); |
293 |
each(children, function(node, i) {
|
294 |
// Jump over last BR element
|
295 |
if (node.nodeName != 'BR' || i != children.length - 1) |
296 |
startCell.appendChild(node); |
297 |
}); |
298 |
|
299 |
// Remove cell
|
300 |
dom.remove(cell); |
301 |
} |
302 |
} |
303 |
} |
304 |
|
305 |
// Remove empty rows etc and restore caret location
|
306 |
cleanup(); |
307 |
} |
308 |
}; |
309 |
|
310 |
function insertRow(before) { |
311 |
var posY, cell, lastCell, x, rowElm, newRow, newCell, otherCell;
|
312 |
|
313 |
// Find first/last row
|
314 |
each(grid, function(row, y) {
|
315 |
each(row, function(cell, x) {
|
316 |
if (isCellSelected(cell)) {
|
317 |
cell = cell.elm; |
318 |
rowElm = cell.parentNode; |
319 |
newRow = cloneNode(rowElm, false);
|
320 |
posY = y; |
321 |
|
322 |
if (before)
|
323 |
return false; |
324 |
} |
325 |
}); |
326 |
|
327 |
if (before)
|
328 |
return !posY;
|
329 |
}); |
330 |
|
331 |
for (x = 0; x < grid[0].length; x++) { |
332 |
cell = grid[posY][x].elm; |
333 |
|
334 |
if (cell != lastCell) {
|
335 |
if (!before) {
|
336 |
rowSpan = getSpanVal(cell, 'rowspan');
|
337 |
if (rowSpan > 1) { |
338 |
cell.rowSpan = rowSpan + 1;
|
339 |
continue;
|
340 |
} |
341 |
} else {
|
342 |
// Check if cell above can be expanded
|
343 |
if (posY > 0 && grid[posY - 1][x]) { |
344 |
otherCell = grid[posY - 1][x].elm;
|
345 |
rowSpan = getSpanVal(otherCell, 'rowspan');
|
346 |
if (rowSpan > 1) { |
347 |
otherCell.rowSpan = rowSpan + 1;
|
348 |
continue;
|
349 |
} |
350 |
} |
351 |
} |
352 |
|
353 |
// Insert new cell into new row
|
354 |
newCell = cloneCell(cell) |
355 |
newCell.colSpan = cell.colSpan; |
356 |
newRow.appendChild(newCell); |
357 |
|
358 |
lastCell = cell; |
359 |
} |
360 |
} |
361 |
|
362 |
if (newRow.hasChildNodes()) {
|
363 |
if (!before)
|
364 |
dom.insertAfter(newRow, rowElm); |
365 |
else
|
366 |
rowElm.parentNode.insertBefore(newRow, rowElm); |
367 |
} |
368 |
}; |
369 |
|
370 |
function insertCol(before) { |
371 |
var posX, lastCell;
|
372 |
|
373 |
// Find first/last column
|
374 |
each(grid, function(row, y) {
|
375 |
each(row, function(cell, x) {
|
376 |
if (isCellSelected(cell)) {
|
377 |
posX = x; |
378 |
|
379 |
if (before)
|
380 |
return false; |
381 |
} |
382 |
}); |
383 |
|
384 |
if (before)
|
385 |
return !posX;
|
386 |
}); |
387 |
|
388 |
each(grid, function(row, y) {
|
389 |
var cell = row[posX].elm, rowSpan, colSpan;
|
390 |
|
391 |
if (cell != lastCell) {
|
392 |
colSpan = getSpanVal(cell, 'colspan');
|
393 |
rowSpan = getSpanVal(cell, 'rowspan');
|
394 |
|
395 |
if (colSpan == 1) { |
396 |
if (!before) {
|
397 |
dom.insertAfter(cloneCell(cell), cell); |
398 |
fillLeftDown(posX, y, rowSpan - 1, colSpan);
|
399 |
} else {
|
400 |
cell.parentNode.insertBefore(cloneCell(cell), cell); |
401 |
fillLeftDown(posX, y, rowSpan - 1, colSpan);
|
402 |
} |
403 |
} else
|
404 |
cell.colSpan++; |
405 |
|
406 |
lastCell = cell; |
407 |
} |
408 |
}); |
409 |
}; |
410 |
|
411 |
function deleteCols() { |
412 |
var cols = [];
|
413 |
|
414 |
// Get selected column indexes
|
415 |
each(grid, function(row, y) {
|
416 |
each(row, function(cell, x) {
|
417 |
if (isCellSelected(cell) && tinymce.inArray(cols, x) === -1) { |
418 |
each(grid, function(row) {
|
419 |
var cell = row[x].elm, colSpan;
|
420 |
|
421 |
colSpan = getSpanVal(cell, 'colspan');
|
422 |
|
423 |
if (colSpan > 1) |
424 |
cell.colSpan = colSpan - 1;
|
425 |
else
|
426 |
dom.remove(cell); |
427 |
}); |
428 |
|
429 |
cols.push(x); |
430 |
} |
431 |
}); |
432 |
}); |
433 |
|
434 |
cleanup(); |
435 |
}; |
436 |
|
437 |
function deleteRows() { |
438 |
var rows;
|
439 |
|
440 |
function deleteRow(tr) { |
441 |
var nextTr, pos, lastCell;
|
442 |
|
443 |
nextTr = dom.getNext(tr, 'tr');
|
444 |
|
445 |
// Move down row spanned cells
|
446 |
each(tr.cells, function(cell) {
|
447 |
var rowSpan = getSpanVal(cell, 'rowspan'); |
448 |
|
449 |
if (rowSpan > 1) { |
450 |
cell.rowSpan = rowSpan - 1;
|
451 |
pos = getPos(cell); |
452 |
fillLeftDown(pos.x, pos.y, 1, 1); |
453 |
} |
454 |
}); |
455 |
|
456 |
// Delete cells
|
457 |
pos = getPos(tr.cells[0]);
|
458 |
each(grid[pos.y], function(cell) {
|
459 |
var rowSpan;
|
460 |
|
461 |
cell = cell.elm; |
462 |
|
463 |
if (cell != lastCell) {
|
464 |
rowSpan = getSpanVal(cell, 'rowspan');
|
465 |
|
466 |
if (rowSpan <= 1) |
467 |
dom.remove(cell); |
468 |
else
|
469 |
cell.rowSpan = rowSpan - 1;
|
470 |
|
471 |
lastCell = cell; |
472 |
} |
473 |
}); |
474 |
}; |
475 |
|
476 |
// Get selected rows and move selection out of scope
|
477 |
rows = getSelectedRows(); |
478 |
|
479 |
// Delete all selected rows
|
480 |
each(rows.reverse(), function(tr) {
|
481 |
deleteRow(tr); |
482 |
}); |
483 |
|
484 |
cleanup(); |
485 |
}; |
486 |
|
487 |
function cutRows() { |
488 |
var rows = getSelectedRows();
|
489 |
|
490 |
dom.remove(rows); |
491 |
cleanup(); |
492 |
|
493 |
return rows;
|
494 |
}; |
495 |
|
496 |
function copyRows() { |
497 |
var rows = getSelectedRows();
|
498 |
|
499 |
each(rows, function(row, i) {
|
500 |
rows[i] = cloneNode(row, true);
|
501 |
}); |
502 |
|
503 |
return rows;
|
504 |
}; |
505 |
|
506 |
function pasteRows(rows, before) { |
507 |
var selectedRows = getSelectedRows(),
|
508 |
targetRow = selectedRows[before ? 0 : selectedRows.length - 1], |
509 |
targetCellCount = targetRow.cells.length; |
510 |
|
511 |
// Calc target cell count
|
512 |
each(grid, function(row) {
|
513 |
var match;
|
514 |
|
515 |
targetCellCount = 0;
|
516 |
each(row, function(cell, x) {
|
517 |
if (cell.real)
|
518 |
targetCellCount += cell.colspan; |
519 |
|
520 |
if (cell.elm.parentNode == targetRow)
|
521 |
match = 1;
|
522 |
}); |
523 |
|
524 |
if (match)
|
525 |
return false; |
526 |
}); |
527 |
|
528 |
if (!before)
|
529 |
rows.reverse(); |
530 |
|
531 |
each(rows, function(row) {
|
532 |
var cellCount = row.cells.length, cell;
|
533 |
|
534 |
// Remove col/rowspans
|
535 |
for (i = 0; i < cellCount; i++) { |
536 |
cell = row.cells[i]; |
537 |
cell.colSpan = cell.rowSpan = 1;
|
538 |
} |
539 |
|
540 |
// Needs more cells
|
541 |
for (i = cellCount; i < targetCellCount; i++)
|
542 |
row.appendChild(cloneCell(row.cells[cellCount - 1]));
|
543 |
|
544 |
// Needs less cells
|
545 |
for (i = targetCellCount; i < cellCount; i++)
|
546 |
dom.remove(row.cells[i]); |
547 |
|
548 |
// Add before/after
|
549 |
if (before)
|
550 |
targetRow.parentNode.insertBefore(row, targetRow); |
551 |
else
|
552 |
dom.insertAfter(row, targetRow); |
553 |
}); |
554 |
}; |
555 |
|
556 |
function getPos(target) { |
557 |
var pos;
|
558 |
|
559 |
each(grid, function(row, y) {
|
560 |
each(row, function(cell, x) {
|
561 |
if (cell.elm == target) {
|
562 |
pos = {x : x, y : y}; |
563 |
return false; |
564 |
} |
565 |
}); |
566 |
|
567 |
return !pos;
|
568 |
}); |
569 |
|
570 |
return pos;
|
571 |
}; |
572 |
|
573 |
function setStartCell(cell) { |
574 |
startPos = getPos(cell); |
575 |
}; |
576 |
|
577 |
function findEndPos() { |
578 |
var pos, maxX, maxY;
|
579 |
|
580 |
maxX = maxY = 0;
|
581 |
|
582 |
each(grid, function(row, y) {
|
583 |
each(row, function(cell, x) {
|
584 |
var colSpan, rowSpan;
|
585 |
|
586 |
if (isCellSelected(cell)) {
|
587 |
cell = grid[y][x]; |
588 |
|
589 |
if (x > maxX)
|
590 |
maxX = x; |
591 |
|
592 |
if (y > maxY)
|
593 |
maxY = y; |
594 |
|
595 |
if (cell.real) {
|
596 |
colSpan = cell.colspan - 1;
|
597 |
rowSpan = cell.rowspan - 1;
|
598 |
|
599 |
if (colSpan) {
|
600 |
if (x + colSpan > maxX)
|
601 |
maxX = x + colSpan; |
602 |
} |
603 |
|
604 |
if (rowSpan) {
|
605 |
if (y + rowSpan > maxY)
|
606 |
maxY = y + rowSpan; |
607 |
} |
608 |
} |
609 |
} |
610 |
}); |
611 |
}); |
612 |
|
613 |
return {x : maxX, y : maxY}; |
614 |
}; |
615 |
|
616 |
function setEndCell(cell) { |
617 |
var startX, startY, endX, endY, maxX, maxY, colSpan, rowSpan;
|
618 |
|
619 |
endPos = getPos(cell); |
620 |
|
621 |
if (startPos && endPos) {
|
622 |
// Get start/end positions
|
623 |
startX = Math.min(startPos.x, endPos.x); |
624 |
startY = Math.min(startPos.y, endPos.y); |
625 |
endX = Math.max(startPos.x, endPos.x); |
626 |
endY = Math.max(startPos.y, endPos.y); |
627 |
|
628 |
// Expand end positon to include spans
|
629 |
maxX = endX; |
630 |
maxY = endY; |
631 |
|
632 |
// Expand startX
|
633 |
for (y = startY; y <= maxY; y++) {
|
634 |
cell = grid[y][startX]; |
635 |
|
636 |
if (!cell.real) {
|
637 |
if (startX - (cell.colspan - 1) < startX) |
638 |
startX -= cell.colspan - 1;
|
639 |
} |
640 |
} |
641 |
|
642 |
// Expand startY
|
643 |
for (x = startX; x <= maxX; x++) {
|
644 |
cell = grid[startY][x]; |
645 |
|
646 |
if (!cell.real) {
|
647 |
if (startY - (cell.rowspan - 1) < startY) |
648 |
startY -= cell.rowspan - 1;
|
649 |
} |
650 |
} |
651 |
|
652 |
// Find max X, Y
|
653 |
for (y = startY; y <= endY; y++) {
|
654 |
for (x = startX; x <= endX; x++) {
|
655 |
cell = grid[y][x]; |
656 |
|
657 |
if (cell.real) {
|
658 |
colSpan = cell.colspan - 1;
|
659 |
rowSpan = cell.rowspan - 1;
|
660 |
|
661 |
if (colSpan) {
|
662 |
if (x + colSpan > maxX)
|
663 |
maxX = x + colSpan; |
664 |
} |
665 |
|
666 |
if (rowSpan) {
|
667 |
if (y + rowSpan > maxY)
|
668 |
maxY = y + rowSpan; |
669 |
} |
670 |
} |
671 |
} |
672 |
} |
673 |
|
674 |
// Remove current selection
|
675 |
dom.removeClass(dom.select('td.mceSelected,th.mceSelected'), 'mceSelected'); |
676 |
|
677 |
// Add new selection
|
678 |
for (y = startY; y <= maxY; y++) {
|
679 |
for (x = startX; x <= maxX; x++)
|
680 |
dom.addClass(grid[y][x].elm, 'mceSelected');
|
681 |
} |
682 |
} |
683 |
}; |
684 |
|
685 |
// Expose to public
|
686 |
tinymce.extend(this, {
|
687 |
deleteTable : deleteTable,
|
688 |
split : split,
|
689 |
merge : merge,
|
690 |
insertRow : insertRow,
|
691 |
insertCol : insertCol,
|
692 |
deleteCols : deleteCols,
|
693 |
deleteRows : deleteRows,
|
694 |
cutRows : cutRows,
|
695 |
copyRows : copyRows,
|
696 |
pasteRows : pasteRows,
|
697 |
getPos : getPos,
|
698 |
setStartCell : setStartCell,
|
699 |
setEndCell : setEndCell
|
700 |
}); |
701 |
}; |
702 |
|
703 |
tinymce.create('tinymce.plugins.TablePlugin', {
|
704 |
init : function(ed, url) { |
705 |
var winMan, clipboardRows;
|
706 |
|
707 |
function createTableGrid(node) { |
708 |
var selection = ed.selection, tblElm = ed.dom.getParent(node || selection.getNode(), 'table'); |
709 |
|
710 |
if (tblElm)
|
711 |
return new TableGrid(tblElm, ed.dom, selection); |
712 |
}; |
713 |
|
714 |
function cleanup() { |
715 |
// Restore selection possibilities
|
716 |
ed.getBody().style.webkitUserSelect = '';
|
717 |
ed.dom.removeClass(ed.dom.select('td.mceSelected,th.mceSelected'), 'mceSelected'); |
718 |
}; |
719 |
|
720 |
// Register buttons
|
721 |
each([ |
722 |
['table', 'table.desc', 'mceInsertTable', true], |
723 |
['delete_table', 'table.del', 'mceTableDelete'], |
724 |
['delete_col', 'table.delete_col_desc', 'mceTableDeleteCol'], |
725 |
['delete_row', 'table.delete_row_desc', 'mceTableDeleteRow'], |
726 |
['col_after', 'table.col_after_desc', 'mceTableInsertColAfter'], |
727 |
['col_before', 'table.col_before_desc', 'mceTableInsertColBefore'], |
728 |
['row_after', 'table.row_after_desc', 'mceTableInsertRowAfter'], |
729 |
['row_before', 'table.row_before_desc', 'mceTableInsertRowBefore'], |
730 |
['row_props', 'table.row_desc', 'mceTableRowProps', true], |
731 |
['cell_props', 'table.cell_desc', 'mceTableCellProps', true], |
732 |
['split_cells', 'table.split_cells_desc', 'mceTableSplitCells', true], |
733 |
['merge_cells', 'table.merge_cells_desc', 'mceTableMergeCells', true] |
734 |
], function(c) {
|
735 |
ed.addButton(c[0], {title : c[1], cmd : c[2], ui : c[3]}); |
736 |
}); |
737 |
|
738 |
// Select whole table is a table border is clicked
|
739 |
if (!tinymce.isIE) {
|
740 |
ed.onClick.add(function(ed, e) {
|
741 |
e = e.target; |
742 |
|
743 |
if (e.nodeName === 'TABLE') |
744 |
ed.selection.select(e); |
745 |
}); |
746 |
} |
747 |
|
748 |
// Handle node change updates
|
749 |
ed.onNodeChange.add(function(ed, cm, n) {
|
750 |
var p;
|
751 |
|
752 |
n = ed.selection.getStart(); |
753 |
p = ed.dom.getParent(n, 'td,th,caption');
|
754 |
cm.setActive('table', n.nodeName === 'TABLE' || !!p); |
755 |
|
756 |
// Disable table tools if we are in caption
|
757 |
if (p && p.nodeName === 'CAPTION') |
758 |
p = 0;
|
759 |
|
760 |
cm.setDisabled('delete_table', !p);
|
761 |
cm.setDisabled('delete_col', !p);
|
762 |
cm.setDisabled('delete_table', !p);
|
763 |
cm.setDisabled('delete_row', !p);
|
764 |
cm.setDisabled('col_after', !p);
|
765 |
cm.setDisabled('col_before', !p);
|
766 |
cm.setDisabled('row_after', !p);
|
767 |
cm.setDisabled('row_before', !p);
|
768 |
cm.setDisabled('row_props', !p);
|
769 |
cm.setDisabled('cell_props', !p);
|
770 |
cm.setDisabled('split_cells', !p);
|
771 |
cm.setDisabled('merge_cells', !p);
|
772 |
}); |
773 |
|
774 |
ed.onInit.add(function(ed) {
|
775 |
var startTable, startCell, dom = ed.dom, tableGrid;
|
776 |
|
777 |
winMan = ed.windowManager; |
778 |
|
779 |
// Add cell selection logic
|
780 |
ed.onMouseDown.add(function(ed, e) {
|
781 |
if (e.button != 2) { |
782 |
cleanup(); |
783 |
|
784 |
startCell = dom.getParent(e.target, 'td,th');
|
785 |
startTable = dom.getParent(startCell, 'table');
|
786 |
} |
787 |
}); |
788 |
|
789 |
dom.bind(ed.getDoc(), 'mouseover', function(e) { |
790 |
var sel, table, target = e.target;
|
791 |
|
792 |
if (startCell && (tableGrid || target != startCell) && (target.nodeName == 'TD' || target.nodeName == 'TH')) { |
793 |
table = dom.getParent(target, 'table');
|
794 |
if (table == startTable) {
|
795 |
if (!tableGrid) {
|
796 |
tableGrid = createTableGrid(table); |
797 |
tableGrid.setStartCell(startCell); |
798 |
|
799 |
ed.getBody().style.webkitUserSelect = 'none';
|
800 |
} |
801 |
|
802 |
tableGrid.setEndCell(target); |
803 |
} |
804 |
|
805 |
// Remove current selection
|
806 |
sel = ed.selection.getSel(); |
807 |
|
808 |
if (sel.removeAllRanges)
|
809 |
sel.removeAllRanges(); |
810 |
else
|
811 |
sel.empty(); |
812 |
|
813 |
e.preventDefault(); |
814 |
} |
815 |
}); |
816 |
|
817 |
ed.onMouseUp.add(function(ed, e) {
|
818 |
var rng, sel = ed.selection, selectedCells, nativeSel = sel.getSel(), walker, node, lastNode, endNode;
|
819 |
|
820 |
// Move selection to startCell
|
821 |
if (startCell) {
|
822 |
if (tableGrid)
|
823 |
ed.getBody().style.webkitUserSelect = '';
|
824 |
|
825 |
function setPoint(node, start) { |
826 |
var walker = new tinymce.dom.TreeWalker(node, node); |
827 |
|
828 |
do {
|
829 |
// Text node
|
830 |
if (node.nodeType == 3 && tinymce.trim(node.nodeValue).length != 0) { |
831 |
if (start)
|
832 |
rng.setStart(node, 0);
|
833 |
else
|
834 |
rng.setEnd(node, node.nodeValue.length); |
835 |
|
836 |
return;
|
837 |
} |
838 |
|
839 |
// BR element
|
840 |
if (node.nodeName == 'BR') { |
841 |
if (start)
|
842 |
rng.setStartBefore(node); |
843 |
else
|
844 |
rng.setEndBefore(node); |
845 |
|
846 |
return;
|
847 |
} |
848 |
} while (node = (start ? walker.next() : walker.prev()));
|
849 |
}; |
850 |
|
851 |
// Try to expand text selection as much as we can only Gecko supports cell selection
|
852 |
selectedCells = dom.select('td.mceSelected,th.mceSelected');
|
853 |
if (selectedCells.length > 0) { |
854 |
rng = dom.createRng(); |
855 |
node = selectedCells[0];
|
856 |
endNode = selectedCells[selectedCells.length - 1];
|
857 |
|
858 |
setPoint(node, 1);
|
859 |
walker = new tinymce.dom.TreeWalker(node, dom.getParent(selectedCells[0], 'table')); |
860 |
|
861 |
do {
|
862 |
if (node.nodeName == 'TD' || node.nodeName == 'TH') { |
863 |
if (!dom.hasClass(node, 'mceSelected')) |
864 |
break;
|
865 |
|
866 |
lastNode = node; |
867 |
} |
868 |
} while (node = walker.next());
|
869 |
|
870 |
setPoint(lastNode); |
871 |
|
872 |
sel.setRng(rng); |
873 |
} |
874 |
|
875 |
ed.nodeChanged(); |
876 |
startCell = tableGrid = startTable = null;
|
877 |
} |
878 |
}); |
879 |
|
880 |
ed.onKeyUp.add(function(ed, e) {
|
881 |
cleanup(); |
882 |
}); |
883 |
|
884 |
// Add context menu
|
885 |
if (ed && ed.plugins.contextmenu) {
|
886 |
ed.plugins.contextmenu.onContextMenu.add(function(th, m, e) {
|
887 |
var sm, se = ed.selection, el = se.getNode() || ed.getBody();
|
888 |
|
889 |
if (ed.dom.getParent(e, 'td') || ed.dom.getParent(e, 'th') || ed.dom.select('td.mceSelected,th.mceSelected').length) { |
890 |
m.removeAll(); |
891 |
|
892 |
if (el.nodeName == 'A' && !ed.dom.getAttrib(el, 'name')) { |
893 |
m.add({title : 'advanced.link_desc', icon : 'link', cmd : ed.plugins.advlink ? 'mceAdvLink' : 'mceLink', ui : true}); |
894 |
m.add({title : 'advanced.unlink_desc', icon : 'unlink', cmd : 'UnLink'}); |
895 |
m.addSeparator(); |
896 |
} |
897 |
|
898 |
if (el.nodeName == 'IMG' && el.className.indexOf('mceItem') == -1) { |
899 |
m.add({title : 'advanced.image_desc', icon : 'image', cmd : ed.plugins.advimage ? 'mceAdvImage' : 'mceImage', ui : true}); |
900 |
m.addSeparator(); |
901 |
} |
902 |
|
903 |
m.add({title : 'table.desc', icon : 'table', cmd : 'mceInsertTable', value : {action : 'insert'}}); |
904 |
m.add({title : 'table.props_desc', icon : 'table_props', cmd : 'mceInsertTable'}); |
905 |
m.add({title : 'table.del', icon : 'delete_table', cmd : 'mceTableDelete'}); |
906 |
m.addSeparator(); |
907 |
|
908 |
// Cell menu
|
909 |
sm = m.addMenu({title : 'table.cell'}); |
910 |
sm.add({title : 'table.cell_desc', icon : 'cell_props', cmd : 'mceTableCellProps'}); |
911 |
sm.add({title : 'table.split_cells_desc', icon : 'split_cells', cmd : 'mceTableSplitCells'}); |
912 |
sm.add({title : 'table.merge_cells_desc', icon : 'merge_cells', cmd : 'mceTableMergeCells'}); |
913 |
|
914 |
// Row menu
|
915 |
sm = m.addMenu({title : 'table.row'}); |
916 |
sm.add({title : 'table.row_desc', icon : 'row_props', cmd : 'mceTableRowProps'}); |
917 |
sm.add({title : 'table.row_before_desc', icon : 'row_before', cmd : 'mceTableInsertRowBefore'}); |
918 |
sm.add({title : 'table.row_after_desc', icon : 'row_after', cmd : 'mceTableInsertRowAfter'}); |
919 |
sm.add({title : 'table.delete_row_desc', icon : 'delete_row', cmd : 'mceTableDeleteRow'}); |
920 |
sm.addSeparator(); |
921 |
sm.add({title : 'table.cut_row_desc', icon : 'cut', cmd : 'mceTableCutRow'}); |
922 |
sm.add({title : 'table.copy_row_desc', icon : 'copy', cmd : 'mceTableCopyRow'}); |
923 |
sm.add({title : 'table.paste_row_before_desc', icon : 'paste', cmd : 'mceTablePasteRowBefore'}).setDisabled(!clipboardRows); |
924 |
sm.add({title : 'table.paste_row_after_desc', icon : 'paste', cmd : 'mceTablePasteRowAfter'}).setDisabled(!clipboardRows); |
925 |
|
926 |
// Column menu
|
927 |
sm = m.addMenu({title : 'table.col'}); |
928 |
sm.add({title : 'table.col_before_desc', icon : 'col_before', cmd : 'mceTableInsertColBefore'}); |
929 |
sm.add({title : 'table.col_after_desc', icon : 'col_after', cmd : 'mceTableInsertColAfter'}); |
930 |
sm.add({title : 'table.delete_col_desc', icon : 'delete_col', cmd : 'mceTableDeleteCol'}); |
931 |
} else
|
932 |
m.add({title : 'table.desc', icon : 'table', cmd : 'mceInsertTable'}); |
933 |
}); |
934 |
} |
935 |
|
936 |
// Fixes an issue on Gecko where it's impossible to place the caret behind a table
|
937 |
// This fix will force a paragraph element after the table but only when the forced_root_block setting is enabled
|
938 |
if (!tinymce.isIE) {
|
939 |
function fixTableCaretPos() { |
940 |
var last;
|
941 |
|
942 |
// Skip empty text nodes form the end
|
943 |
for (last = ed.getBody().lastChild; last && last.nodeType == 3 && !last.nodeValue.length; last = last.previousSibling) ; |
944 |
|
945 |
if (last && last.nodeName == 'TABLE') |
946 |
ed.dom.add(ed.getBody(), 'p', null, '<br mce_bogus="1" />'); |
947 |
}; |
948 |
|
949 |
// Fixes an bug where it's impossible to place the caret before a table in Gecko
|
950 |
// this fix solves it by detecting when the caret is at the beginning of such a table
|
951 |
// and then manually moves the caret infront of the table
|
952 |
if (tinymce.isGecko) {
|
953 |
ed.onKeyDown.add(function(ed, e) {
|
954 |
var rng, table, dom = ed.dom;
|
955 |
|
956 |
// On gecko it's not possible to place the caret before a table
|
957 |
if (e.keyCode == 37 || e.keyCode == 38) { |
958 |
rng = ed.selection.getRng(); |
959 |
table = dom.getParent(rng.startContainer, 'table');
|
960 |
|
961 |
if (table && ed.getBody().firstChild == table) {
|
962 |
if (isAtStart(rng, table)) {
|
963 |
rng = dom.createRng(); |
964 |
|
965 |
rng.setStartBefore(table); |
966 |
rng.setEndBefore(table); |
967 |
|
968 |
ed.selection.setRng(rng); |
969 |
|
970 |
e.preventDefault(); |
971 |
} |
972 |
} |
973 |
} |
974 |
}); |
975 |
} |
976 |
|
977 |
ed.onKeyUp.add(fixTableCaretPos); |
978 |
ed.onSetContent.add(fixTableCaretPos); |
979 |
ed.onVisualAid.add(fixTableCaretPos); |
980 |
|
981 |
ed.onPreProcess.add(function(ed, o) {
|
982 |
var last = o.node.lastChild;
|
983 |
|
984 |
if (last && last.childNodes.length == 1 && last.firstChild.nodeName == 'BR') |
985 |
ed.dom.remove(last); |
986 |
}); |
987 |
|
988 |
fixTableCaretPos(); |
989 |
} |
990 |
}); |
991 |
|
992 |
// Register action commands
|
993 |
each({ |
994 |
mceTableSplitCells : function(grid) { |
995 |
grid.split(); |
996 |
}, |
997 |
|
998 |
mceTableMergeCells : function(grid) { |
999 |
var rowSpan, colSpan, cell;
|
1000 |
|
1001 |
cell = ed.dom.getParent(ed.selection.getNode(), 'th,td');
|
1002 |
if (cell) {
|
1003 |
rowSpan = cell.rowSpan; |
1004 |
colSpan = cell.colSpan; |
1005 |
} |
1006 |
|
1007 |
if (!ed.dom.select('td.mceSelected,th.mceSelected').length) { |
1008 |
winMan.open({ |
1009 |
url : url + '/merge_cells.htm', |
1010 |
width : 240 + parseInt(ed.getLang('table.merge_cells_delta_width', 0)), |
1011 |
height : 110 + parseInt(ed.getLang('table.merge_cells_delta_height', 0)), |
1012 |
inline : 1 |
1013 |
}, { |
1014 |
rows : rowSpan,
|
1015 |
cols : colSpan,
|
1016 |
onaction : function(data) { |
1017 |
grid.merge(cell, data.cols, data.rows); |
1018 |
}, |
1019 |
plugin_url : url
|
1020 |
}); |
1021 |
} else
|
1022 |
grid.merge(); |
1023 |
}, |
1024 |
|
1025 |
mceTableInsertRowBefore : function(grid) { |
1026 |
grid.insertRow(true);
|
1027 |
}, |
1028 |
|
1029 |
mceTableInsertRowAfter : function(grid) { |
1030 |
grid.insertRow(); |
1031 |
}, |
1032 |
|
1033 |
mceTableInsertColBefore : function(grid) { |
1034 |
grid.insertCol(true);
|
1035 |
}, |
1036 |
|
1037 |
mceTableInsertColAfter : function(grid) { |
1038 |
grid.insertCol(); |
1039 |
}, |
1040 |
|
1041 |
mceTableDeleteCol : function(grid) { |
1042 |
grid.deleteCols(); |
1043 |
}, |
1044 |
|
1045 |
mceTableDeleteRow : function(grid) { |
1046 |
grid.deleteRows(); |
1047 |
}, |
1048 |
|
1049 |
mceTableCutRow : function(grid) { |
1050 |
clipboardRows = grid.cutRows(); |
1051 |
}, |
1052 |
|
1053 |
mceTableCopyRow : function(grid) { |
1054 |
clipboardRows = grid.copyRows(); |
1055 |
}, |
1056 |
|
1057 |
mceTablePasteRowBefore : function(grid) { |
1058 |
grid.pasteRows(clipboardRows, true);
|
1059 |
}, |
1060 |
|
1061 |
mceTablePasteRowAfter : function(grid) { |
1062 |
grid.pasteRows(clipboardRows); |
1063 |
}, |
1064 |
|
1065 |
mceTableDelete : function(grid) { |
1066 |
grid.deleteTable(); |
1067 |
} |
1068 |
}, function(func, name) {
|
1069 |
ed.addCommand(name, function() {
|
1070 |
var grid = createTableGrid();
|
1071 |
|
1072 |
if (grid) {
|
1073 |
func(grid); |
1074 |
ed.execCommand('mceRepaint');
|
1075 |
cleanup(); |
1076 |
} |
1077 |
}); |
1078 |
}); |
1079 |
|
1080 |
// Register dialog commands
|
1081 |
each({ |
1082 |
mceInsertTable : function(val) { |
1083 |
winMan.open({ |
1084 |
url : url + '/table.htm', |
1085 |
width : 400 + parseInt(ed.getLang('table.table_delta_width', 0)), |
1086 |
height : 320 + parseInt(ed.getLang('table.table_delta_height', 0)), |
1087 |
inline : 1 |
1088 |
}, { |
1089 |
plugin_url : url,
|
1090 |
action : val ? val.action : 0 |
1091 |
}); |
1092 |
}, |
1093 |
|
1094 |
mceTableRowProps : function() { |
1095 |
winMan.open({ |
1096 |
url : url + '/row.htm', |
1097 |
width : 400 + parseInt(ed.getLang('table.rowprops_delta_width', 0)), |
1098 |
height : 295 + parseInt(ed.getLang('table.rowprops_delta_height', 0)), |
1099 |
inline : 1 |
1100 |
}, { |
1101 |
plugin_url : url
|
1102 |
}); |
1103 |
}, |
1104 |
|
1105 |
mceTableCellProps : function() { |
1106 |
winMan.open({ |
1107 |
url : url + '/cell.htm', |
1108 |
width : 400 + parseInt(ed.getLang('table.cellprops_delta_width', 0)), |
1109 |
height : 295 + parseInt(ed.getLang('table.cellprops_delta_height', 0)), |
1110 |
inline : 1 |
1111 |
}, { |
1112 |
plugin_url : url
|
1113 |
}); |
1114 |
} |
1115 |
}, function(func, name) {
|
1116 |
ed.addCommand(name, function(ui, val) {
|
1117 |
func(val); |
1118 |
}); |
1119 |
}); |
1120 |
} |
1121 |
}); |
1122 |
|
1123 |
// Register plugin
|
1124 |
tinymce.PluginManager.add('table', tinymce.plugins.TablePlugin);
|
1125 |
})(tinymce); |