fcktablehandler.js
上传用户:dbstep
上传日期:2022-08-06
资源大小:2803k
文件大小:26k
源码类别:

WEB源码(ASP,PHP,...)

开发平台:

ASP/ASPX

  1. /*
  2.  * FCKeditor - The text editor for Internet - http://www.fckeditor.net
  3.  * Copyright (C) 2003-2009 Frederico Caldeira Knabben
  4.  *
  5.  * == BEGIN LICENSE ==
  6.  *
  7.  * Licensed under the terms of any of the following licenses at your
  8.  * choice:
  9.  *
  10.  *  - GNU General Public License Version 2 or later (the "GPL")
  11.  *    http://www.gnu.org/licenses/gpl.html
  12.  *
  13.  *  - GNU Lesser General Public License Version 2.1 or later (the "LGPL")
  14.  *    http://www.gnu.org/licenses/lgpl.html
  15.  *
  16.  *  - Mozilla Public License Version 1.1 or later (the "MPL")
  17.  *    http://www.mozilla.org/MPL/MPL-1.1.html
  18.  *
  19.  * == END LICENSE ==
  20.  *
  21.  * Manage table operations.
  22.  */
  23. var FCKTableHandler = new Object() ;
  24. FCKTableHandler.InsertRow = function( insertBefore )
  25. {
  26. // Get the row where the selection is placed in.
  27. var oRow = FCKSelection.MoveToAncestorNode( 'TR' ) ;
  28. if ( !oRow ) return ;
  29. // Create a clone of the row.
  30. var oNewRow = oRow.cloneNode( true ) ;
  31. // Insert the new row (copy) before of it.
  32. oRow.parentNode.insertBefore( oNewRow, oRow ) ;
  33. // Clean one of the rows to produce the illusion of inserting an empty row before or after.
  34. FCKTableHandler.ClearRow( insertBefore ? oNewRow : oRow ) ;
  35. }
  36. FCKTableHandler.DeleteRows = function( row )
  37. {
  38. // If no row has been passed as a parameter,
  39. // then get the row( s ) containing the cells where the selection is placed in.
  40. // If user selected multiple rows ( by selecting multiple cells ), walk
  41. // the selected cell list and delete the rows containing the selected cells
  42. if ( ! row )
  43. {
  44. var aCells = FCKTableHandler.GetSelectedCells() ;
  45. var aRowsToDelete = new Array() ;
  46. //queue up the rows -- it's possible ( and likely ) that we may get duplicates
  47. for ( var i = 0; i < aCells.length; i++ )
  48. {
  49. var oRow = aCells[i].parentNode ;
  50. aRowsToDelete[oRow.rowIndex] = oRow ;
  51. }
  52. for ( var i = aRowsToDelete.length; i >= 0; i-- )
  53. {
  54. if ( aRowsToDelete[i] )
  55. FCKTableHandler.DeleteRows( aRowsToDelete[i] );
  56. }
  57. return ;
  58. }
  59. // Get the row's table.
  60. var oTable = FCKTools.GetElementAscensor( row, 'TABLE' ) ;
  61. // If just one row is available then delete the entire table.
  62. if ( oTable.rows.length == 1 )
  63. {
  64. FCKTableHandler.DeleteTable( oTable ) ;
  65. return ;
  66. }
  67. // Delete the row.
  68. row.parentNode.removeChild( row ) ;
  69. }
  70. FCKTableHandler.DeleteTable = function( table )
  71. {
  72. // If no table has been passed as a parameter,
  73. // then get the table where the selection is placed in.
  74. if ( !table )
  75. {
  76. table = FCKSelection.GetSelectedElement() ;
  77. if ( !table || table.tagName != 'TABLE' )
  78. table = FCKSelection.MoveToAncestorNode( 'TABLE' ) ;
  79. }
  80. if ( !table ) return ;
  81. // Delete the table.
  82. FCKSelection.SelectNode( table ) ;
  83. FCKSelection.Collapse();
  84. // if the table is wrapped with a singleton <p> ( or something similar ), remove
  85. // the surrounding tag -- which likely won't show after deletion anyway
  86. if ( table.parentNode.childNodes.length == 1 )
  87. table.parentNode.parentNode.removeChild( table.parentNode );
  88. else
  89. table.parentNode.removeChild( table  ) ;
  90. }
  91. FCKTableHandler.InsertColumn = function( insertBefore )
  92. {
  93. // Get the cell where the selection is placed in.
  94. var oCell = null ;
  95. var nodes = this.GetSelectedCells() ;
  96. if ( nodes && nodes.length )
  97. oCell = nodes[ insertBefore ? 0 : ( nodes.length - 1 ) ] ;
  98. if ( ! oCell )
  99. return ;
  100. // Get the cell's table.
  101. var oTable = FCKTools.GetElementAscensor( oCell, 'TABLE' ) ;
  102. var iIndex = oCell.cellIndex ;
  103. // Loop through all rows available in the table.
  104. for ( var i = 0 ; i < oTable.rows.length ; i++ )
  105. {
  106. // Get the row.
  107. var oRow = oTable.rows[i] ;
  108. // If the row doesn't have enough cells, ignore it.
  109. if ( oRow.cells.length < ( iIndex + 1 ) )
  110. continue ;
  111. oCell = oRow.cells[iIndex].cloneNode(false) ;
  112. if ( FCKBrowserInfo.IsGeckoLike )
  113. FCKTools.AppendBogusBr( oCell ) ;
  114. // Get back the currently selected cell.
  115. var oBaseCell = oRow.cells[iIndex] ;
  116. oRow.insertBefore( oCell, ( insertBefore ? oBaseCell : oBaseCell.nextSibling ) ) ;
  117. }
  118. }
  119. FCKTableHandler.DeleteColumns = function( oCell )
  120. {
  121. // if user selected multiple cols ( by selecting multiple cells ), walk
  122. // the selected cell list and delete the rows containing the selected cells
  123. if ( !oCell  )
  124. {
  125. var aColsToDelete = FCKTableHandler.GetSelectedCells();
  126. for ( var i = aColsToDelete.length; i >= 0; i--  )
  127. {
  128. if ( aColsToDelete[i]  )
  129. FCKTableHandler.DeleteColumns( aColsToDelete[i]  );
  130. }
  131. return;
  132. }
  133. if ( !oCell ) return ;
  134. // Get the cell's table.
  135. var oTable = FCKTools.GetElementAscensor( oCell, 'TABLE' ) ;
  136. // Get the cell index.
  137. var iIndex = oCell.cellIndex ;
  138. // Loop throw all rows (from down to up, because it's possible that some
  139. // rows will be deleted).
  140. for ( var i = oTable.rows.length - 1 ; i >= 0 ; i-- )
  141. {
  142. // Get the row.
  143. var oRow = oTable.rows[i] ;
  144. // If the cell to be removed is the first one and the row has just one cell.
  145. if ( iIndex == 0 && oRow.cells.length == 1 )
  146. {
  147. // Remove the entire row.
  148. FCKTableHandler.DeleteRows( oRow ) ;
  149. continue ;
  150. }
  151. // If the cell to be removed exists the delete it.
  152. if ( oRow.cells[iIndex] )
  153. oRow.removeChild( oRow.cells[iIndex] ) ;
  154. }
  155. }
  156. FCKTableHandler.InsertCell = function( cell, insertBefore )
  157. {
  158. // Get the cell where the selection is placed in.
  159. var oCell = null ;
  160. var nodes = this.GetSelectedCells() ;
  161. if ( nodes && nodes.length )
  162. oCell = nodes[ insertBefore ? 0 : ( nodes.length - 1 ) ] ;
  163. if ( ! oCell )
  164. return null ;
  165. // Create the new cell element to be added.
  166. var oNewCell = FCK.EditorDocument.createElement( 'TD' ) ;
  167. if ( FCKBrowserInfo.IsGeckoLike )
  168. FCKTools.AppendBogusBr( oNewCell ) ;
  169. if ( !insertBefore && oCell.cellIndex == oCell.parentNode.cells.length - 1 )
  170. oCell.parentNode.appendChild( oNewCell ) ;
  171. else
  172. oCell.parentNode.insertBefore( oNewCell, insertBefore ? oCell : oCell.nextSibling ) ;
  173. return oNewCell ;
  174. }
  175. FCKTableHandler.DeleteCell = function( cell )
  176. {
  177. // If this is the last cell in the row.
  178. if ( cell.parentNode.cells.length == 1 )
  179. {
  180. // Delete the entire row.
  181. FCKTableHandler.DeleteRows( cell.parentNode ) ;
  182. return ;
  183. }
  184. // Delete the cell from the row.
  185. cell.parentNode.removeChild( cell ) ;
  186. }
  187. FCKTableHandler.DeleteCells = function()
  188. {
  189. var aCells = FCKTableHandler.GetSelectedCells() ;
  190. for ( var i = aCells.length - 1 ; i >= 0  ; i-- )
  191. {
  192. FCKTableHandler.DeleteCell( aCells[i] ) ;
  193. }
  194. }
  195. FCKTableHandler._MarkCells = function( cells, label )
  196. {
  197. for ( var i = 0 ; i < cells.length ; i++ )
  198. cells[i][label] = true ;
  199. }
  200. FCKTableHandler._UnmarkCells = function( cells, label )
  201. {
  202. for ( var i = 0 ; i < cells.length ; i++ )
  203. {
  204. FCKDomTools.ClearElementJSProperty(cells[i], label ) ;
  205. }
  206. }
  207. FCKTableHandler._ReplaceCellsByMarker = function( tableMap, marker, substitute )
  208. {
  209. for ( var i = 0 ; i < tableMap.length ; i++ )
  210. {
  211. for ( var j = 0 ; j < tableMap[i].length ; j++ )
  212. {
  213. if ( tableMap[i][j][marker] )
  214. tableMap[i][j] = substitute ;
  215. }
  216. }
  217. }
  218. FCKTableHandler._GetMarkerGeometry = function( tableMap, rowIdx, colIdx, markerName )
  219. {
  220. var selectionWidth = 0 ;
  221. var selectionHeight = 0 ;
  222. var cellsLeft = 0 ;
  223. var cellsUp = 0 ;
  224. for ( var i = colIdx ; tableMap[rowIdx][i] && tableMap[rowIdx][i][markerName] ; i++ )
  225. selectionWidth++ ;
  226. for ( var i = colIdx - 1 ; tableMap[rowIdx][i] && tableMap[rowIdx][i][markerName] ; i-- )
  227. {
  228. selectionWidth++ ;
  229. cellsLeft++ ;
  230. }
  231. for ( var i = rowIdx ; tableMap[i] && tableMap[i][colIdx] && tableMap[i][colIdx][markerName] ; i++ )
  232. selectionHeight++ ;
  233. for ( var i = rowIdx - 1 ; tableMap[i] && tableMap[i][colIdx] && tableMap[i][colIdx][markerName] ; i-- )
  234. {
  235. selectionHeight++ ;
  236. cellsUp++ ;
  237. }
  238. return { 'width' : selectionWidth, 'height' : selectionHeight, 'x' : cellsLeft, 'y' : cellsUp } ;
  239. }
  240. FCKTableHandler.CheckIsSelectionRectangular = function()
  241. {
  242. // If every row and column in an area on a plane are of the same width and height,
  243. // Then the area is a rectangle.
  244. var cells = FCKTableHandler.GetSelectedCells() ;
  245. if ( cells.length < 1 )
  246. return false ;
  247. // Check if the selected cells are all in the same table section (thead, tfoot or tbody)
  248. for (var i = 0; i < cells.length; i++)
  249. {
  250. if ( cells[i].parentNode.parentNode != cells[0].parentNode.parentNode )
  251. return false ;
  252. }
  253. this._MarkCells( cells, '_CellSelected' ) ;
  254. var tableMap = this._CreateTableMap( cells[0] ) ;
  255. var rowIdx = cells[0].parentNode.rowIndex ;
  256. var colIdx = this._GetCellIndexSpan( tableMap, rowIdx, cells[0] ) ;
  257. var geometry = this._GetMarkerGeometry( tableMap, rowIdx, colIdx, '_CellSelected' ) ;
  258. var baseColIdx = colIdx - geometry.x ;
  259. var baseRowIdx = rowIdx - geometry.y ;
  260. if ( geometry.width >= geometry.height )
  261. {
  262. for ( colIdx = baseColIdx ; colIdx < baseColIdx + geometry.width ; colIdx++ )
  263. {
  264. rowIdx = baseRowIdx + ( colIdx - baseColIdx ) % geometry.height ;
  265. if ( ! tableMap[rowIdx] || ! tableMap[rowIdx][colIdx] )
  266. {
  267. this._UnmarkCells( cells, '_CellSelected' ) ;
  268. return false ;
  269. }
  270. var g = this._GetMarkerGeometry( tableMap, rowIdx, colIdx, '_CellSelected' ) ;
  271. if ( g.width != geometry.width || g.height != geometry.height )
  272. {
  273. this._UnmarkCells( cells, '_CellSelected' ) ;
  274. return false ;
  275. }
  276. }
  277. }
  278. else
  279. {
  280. for ( rowIdx = baseRowIdx ; rowIdx < baseRowIdx + geometry.height ; rowIdx++ )
  281. {
  282. colIdx = baseColIdx + ( rowIdx - baseRowIdx ) % geometry.width ;
  283. if ( ! tableMap[rowIdx] || ! tableMap[rowIdx][colIdx] )
  284. {
  285. this._UnmarkCells( cells, '_CellSelected' ) ;
  286. return false ;
  287. }
  288. var g = this._GetMarkerGeometry( tableMap, rowIdx, colIdx, '_CellSelected' ) ;
  289. if ( g.width != geometry.width || g.height != geometry.height )
  290. {
  291. this._UnmarkCells( cells, '_CellSelected' ) ;
  292. return false ;
  293. }
  294. }
  295. }
  296. this._UnmarkCells( cells, '_CellSelected' ) ;
  297. return true ;
  298. }
  299. FCKTableHandler.MergeCells = function()
  300. {
  301. // Get all selected cells.
  302. var cells = this.GetSelectedCells() ;
  303. if ( cells.length < 2 )
  304. return ;
  305. // Assume the selected cells are already in a rectangular geometry.
  306. // Because the checking is already done by FCKTableCommand.
  307. var refCell = cells[0] ;
  308. var tableMap = this._CreateTableMap( refCell ) ;
  309. var rowIdx = refCell.parentNode.rowIndex ;
  310. var colIdx = this._GetCellIndexSpan( tableMap, rowIdx, refCell ) ;
  311. this._MarkCells( cells, '_SelectedCells' ) ;
  312. var selectionGeometry = this._GetMarkerGeometry( tableMap, rowIdx, colIdx, '_SelectedCells' ) ;
  313. var baseColIdx = colIdx - selectionGeometry.x ;
  314. var baseRowIdx = rowIdx - selectionGeometry.y ;
  315. var cellContents = FCKTools.GetElementDocument( refCell ).createDocumentFragment() ;
  316. for ( var i = 0 ; i < selectionGeometry.height ; i++ )
  317. {
  318. var rowChildNodesCount = 0 ;
  319. for ( var j = 0 ; j < selectionGeometry.width ; j++ )
  320. {
  321. var currentCell = tableMap[baseRowIdx + i][baseColIdx + j] ;
  322. while ( currentCell.childNodes.length > 0 )
  323. {
  324. var node = currentCell.removeChild( currentCell.firstChild ) ;
  325. if ( node.nodeType != 1
  326. || ( node.getAttribute( 'type', 2 ) != '_moz' && node.getAttribute( '_moz_dirty' ) != null ) )
  327. {
  328. cellContents.appendChild( node ) ;
  329. rowChildNodesCount++ ;
  330. }
  331. }
  332. }
  333. if ( rowChildNodesCount > 0 )
  334. cellContents.appendChild( FCK.EditorDocument.createElement( 'br' ) ) ;
  335. }
  336. this._ReplaceCellsByMarker( tableMap, '_SelectedCells', refCell ) ;
  337. this._UnmarkCells( cells, '_SelectedCells' ) ;
  338. this._InstallTableMap( tableMap, refCell.parentNode.parentNode.parentNode ) ;
  339. refCell.appendChild( cellContents ) ;
  340. if ( FCKBrowserInfo.IsGeckoLike && ( ! refCell.firstChild ) )
  341. FCKTools.AppendBogusBr( refCell ) ;
  342. this._MoveCaretToCell( refCell, false ) ;
  343. }
  344. FCKTableHandler.MergeRight = function()
  345. {
  346. var target = this.GetMergeRightTarget() ;
  347. if ( target == null )
  348. return ;
  349. var refCell = target.refCell ;
  350. var tableMap = target.tableMap ;
  351. var nextCell = target.nextCell ;
  352. var cellContents = FCK.EditorDocument.createDocumentFragment() ;
  353. while ( nextCell && nextCell.childNodes && nextCell.childNodes.length > 0 )
  354. cellContents.appendChild( nextCell.removeChild( nextCell.firstChild ) ) ;
  355. nextCell.parentNode.removeChild( nextCell ) ;
  356. refCell.appendChild( cellContents ) ;
  357. this._MarkCells( [nextCell], '_Replace' ) ;
  358. this._ReplaceCellsByMarker( tableMap, '_Replace', refCell ) ;
  359. this._InstallTableMap( tableMap, refCell.parentNode.parentNode.parentNode ) ;
  360. this._MoveCaretToCell( refCell, false ) ;
  361. }
  362. FCKTableHandler.MergeDown = function()
  363. {
  364. var target = this.GetMergeDownTarget() ;
  365. if ( target == null )
  366. return ;
  367. var refCell = target.refCell ;
  368. var tableMap = target.tableMap ;
  369. var nextCell = target.nextCell ;
  370. var cellContents = FCKTools.GetElementDocument( refCell ).createDocumentFragment() ;
  371. while ( nextCell && nextCell.childNodes && nextCell.childNodes.length > 0 )
  372. cellContents.appendChild( nextCell.removeChild( nextCell.firstChild ) ) ;
  373. if ( cellContents.firstChild )
  374. cellContents.insertBefore( FCK.EditorDocument.createElement( 'br' ), cellContents.firstChild ) ;
  375. refCell.appendChild( cellContents ) ;
  376. this._MarkCells( [nextCell], '_Replace' ) ;
  377. this._ReplaceCellsByMarker( tableMap, '_Replace', refCell ) ;
  378. this._InstallTableMap( tableMap, refCell.parentNode.parentNode.parentNode ) ;
  379. this._MoveCaretToCell( refCell, false ) ;
  380. }
  381. FCKTableHandler.HorizontalSplitCell = function()
  382. {
  383. var cells = FCKTableHandler.GetSelectedCells() ;
  384. if ( cells.length != 1 )
  385. return ;
  386. var refCell = cells[0] ;
  387. var tableMap = this._CreateTableMap( refCell ) ;
  388. var rowIdx = refCell.parentNode.rowIndex ;
  389. var colIdx = FCKTableHandler._GetCellIndexSpan( tableMap, rowIdx, refCell ) ;
  390. var cellSpan = isNaN( refCell.colSpan ) ? 1 : refCell.colSpan ;
  391. if ( cellSpan > 1 )
  392. {
  393. // Splitting a multi-column cell - original cell gets ceil(colSpan/2) columns,
  394. // new cell gets floor(colSpan/2).
  395. var newCellSpan = Math.ceil( cellSpan / 2 ) ;
  396. var newCell = FCK.EditorDocument.createElement( refCell.nodeName ) ;
  397. if ( FCKBrowserInfo.IsGeckoLike )
  398. FCKTools.AppendBogusBr( newCell ) ;
  399. var startIdx = colIdx + newCellSpan ;
  400. var endIdx = colIdx + cellSpan ;
  401. var rowSpan = isNaN( refCell.rowSpan ) ? 1 : refCell.rowSpan ;
  402. for ( var r = rowIdx ; r < rowIdx + rowSpan ; r++ )
  403. {
  404. for ( var i = startIdx ; i < endIdx ; i++ )
  405. tableMap[r][i] = newCell ;
  406. }
  407. }
  408. else
  409. {
  410. // Splitting a single-column cell - add a new cell, and expand
  411. // cells crossing the same column.
  412. var newTableMap = [] ;
  413. for ( var i = 0 ; i < tableMap.length ; i++ )
  414. {
  415. var newRow = tableMap[i].slice( 0, colIdx ) ;
  416. if ( tableMap[i].length <= colIdx )
  417. {
  418. newTableMap.push( newRow ) ;
  419. continue ;
  420. }
  421. if ( tableMap[i][colIdx] == refCell )
  422. {
  423. newRow.push( refCell ) ;
  424. newRow.push( FCK.EditorDocument.createElement( refCell.nodeName ) ) ;
  425. if ( FCKBrowserInfo.IsGeckoLike )
  426. FCKTools.AppendBogusBr( newRow[newRow.length - 1] ) ;
  427. }
  428. else
  429. {
  430. newRow.push( tableMap[i][colIdx] ) ;
  431. newRow.push( tableMap[i][colIdx] ) ;
  432. }
  433. for ( var j = colIdx + 1 ; j < tableMap[i].length ; j++ )
  434. newRow.push( tableMap[i][j] ) ;
  435. newTableMap.push( newRow ) ;
  436. }
  437. tableMap = newTableMap ;
  438. }
  439. this._InstallTableMap( tableMap, refCell.parentNode.parentNode.parentNode ) ;
  440. }
  441. FCKTableHandler.VerticalSplitCell = function()
  442. {
  443. var cells = FCKTableHandler.GetSelectedCells() ;
  444. if ( cells.length != 1 )
  445. return ;
  446. var currentCell = cells[0] ;
  447. var tableMap = this._CreateTableMap( currentCell ) ;
  448. var currentRowIndex = currentCell.parentNode.rowIndex ;
  449. var cellIndex = FCKTableHandler._GetCellIndexSpan( tableMap, currentRowIndex, currentCell ) ;
  450. // Save current cell colSpan
  451. var currentColSpan = isNaN( currentCell.colSpan ) ? 1 : currentCell.colSpan ;
  452. var currentRowSpan = currentCell.rowSpan ;
  453. if ( isNaN( currentRowSpan ) )
  454. currentRowSpan = 1 ;
  455. if ( currentRowSpan > 1 )
  456. {
  457. // 1. Set the current cell's rowSpan to 1.
  458. currentCell.rowSpan = Math.ceil( currentRowSpan / 2 ) ;
  459. // 2. Find the appropriate place to insert a new cell at the next row.
  460. var newCellRowIndex = currentRowIndex + Math.ceil( currentRowSpan / 2 ) ;
  461. var oRow = tableMap[newCellRowIndex] ;
  462. var insertMarker = null ;
  463. for ( var i = cellIndex+1 ; i < oRow.length ; i++ )
  464. {
  465. if ( oRow[i].parentNode.rowIndex == newCellRowIndex )
  466. {
  467. insertMarker = oRow[i] ;
  468. break ;
  469. }
  470. }
  471. // 3. Insert the new cell to the indicated place, with the appropriate rowSpan and colSpan, next row.
  472. var newCell = FCK.EditorDocument.createElement( currentCell.nodeName ) ;
  473. newCell.rowSpan = Math.floor( currentRowSpan / 2 ) ;
  474. if ( currentColSpan > 1 )
  475. newCell.colSpan = currentColSpan ;
  476. if ( FCKBrowserInfo.IsGeckoLike )
  477. FCKTools.AppendBogusBr( newCell ) ;
  478. currentCell.parentNode.parentNode.parentNode.rows[newCellRowIndex].insertBefore( newCell, insertMarker ) ;
  479. }
  480. else
  481. {
  482. // 1. Insert a new row.
  483. var newSectionRowIdx = currentCell.parentNode.sectionRowIndex + 1 ;
  484. var newRow = FCK.EditorDocument.createElement( 'tr' ) ;
  485. var tSection = currentCell.parentNode.parentNode ;
  486. if ( tSection.rows.length > newSectionRowIdx )
  487. tSection.insertBefore( newRow, tSection.rows[newSectionRowIdx] ) ;
  488. else
  489. tSection.appendChild( newRow ) ;
  490. // 2. +1 to rowSpan for all cells crossing currentCell's row.
  491. for ( var i = 0 ; i < tableMap[currentRowIndex].length ; )
  492. {
  493. var colSpan = tableMap[currentRowIndex][i].colSpan ;
  494. if ( isNaN( colSpan ) || colSpan < 1 )
  495. colSpan = 1 ;
  496. if ( i == cellIndex )
  497. {
  498. i += colSpan ;
  499. continue ;
  500. }
  501. var rowSpan = tableMap[currentRowIndex][i].rowSpan ;
  502. if ( isNaN( rowSpan ) )
  503. rowSpan = 1 ;
  504. tableMap[currentRowIndex][i].rowSpan = rowSpan + 1 ;
  505. i += colSpan ;
  506. }
  507. // 3. Insert a new cell to new row. Set colSpan on the new cell.
  508. var newCell = FCK.EditorDocument.createElement( currentCell.nodeName ) ;
  509. if ( currentColSpan > 1 )
  510. newCell.colSpan = currentColSpan ;
  511. if ( FCKBrowserInfo.IsGeckoLike )
  512. FCKTools.AppendBogusBr( newCell ) ;
  513. newRow.appendChild( newCell ) ;
  514. }
  515. }
  516. // Get the cell index from a TableMap.
  517. FCKTableHandler._GetCellIndexSpan = function( tableMap, rowIndex, cell )
  518. {
  519. if ( tableMap.length < rowIndex + 1 )
  520. return null ;
  521. var oRow = tableMap[ rowIndex ] ;
  522. for ( var c = 0 ; c < oRow.length ; c++ )
  523. {
  524. if ( oRow[c] == cell )
  525. return c ;
  526. }
  527. return null ;
  528. }
  529. // Get the cell location from a TableMap. Returns an array with an [x,y] location
  530. FCKTableHandler._GetCellLocation = function( tableMap, cell  )
  531. {
  532. for ( var i = 0 ; i < tableMap.length; i++ )
  533. {
  534. for ( var c = 0 ; c < tableMap[i].length ; c++  )
  535. {
  536. if ( tableMap[i][c] == cell  ) return [i,c];
  537. }
  538. }
  539. return null ;
  540. }
  541. // This function is quite hard to explain. It creates a matrix representing all cells in a table.
  542. // The difference here is that the "spanned" cells (colSpan and rowSpan) are duplicated on the matrix
  543. // cells that are "spanned". For example, a row with 3 cells where the second cell has colSpan=2 and rowSpan=3
  544. // will produce a bi-dimensional matrix with the following values (representing the cells):
  545. // Cell1, Cell2, Cell2, Cell3
  546. // Cell4, Cell2, Cell2, Cell5
  547. // Cell6, Cell2, Cell2, Cell7
  548. FCKTableHandler._CreateTableMap = function( refCell )
  549. {
  550. var table = (refCell.nodeName == 'TABLE' ? refCell : refCell.parentNode.parentNode.parentNode ) ;
  551. var aRows = table.rows ;
  552. // Row and Column counters.
  553. var r = -1 ;
  554. var aMap = new Array() ;
  555. for ( var i = 0 ; i < aRows.length ; i++ )
  556. {
  557. r++ ;
  558. if ( !aMap[r] )
  559. aMap[r] = new Array() ;
  560. var c = -1 ;
  561. for ( var j = 0 ; j < aRows[i].cells.length ; j++ )
  562. {
  563. var oCell = aRows[i].cells[j] ;
  564. c++ ;
  565. while ( aMap[r][c] )
  566. c++ ;
  567. var iColSpan = isNaN( oCell.colSpan ) ? 1 : oCell.colSpan ;
  568. var iRowSpan = isNaN( oCell.rowSpan ) ? 1 : oCell.rowSpan ;
  569. for ( var rs = 0 ; rs < iRowSpan ; rs++ )
  570. {
  571. if ( !aMap[r + rs] )
  572. aMap[r + rs] = new Array() ;
  573. for ( var cs = 0 ; cs < iColSpan ; cs++ )
  574. {
  575. aMap[r + rs][c + cs] = aRows[i].cells[j] ;
  576. }
  577. }
  578. c += iColSpan - 1 ;
  579. }
  580. }
  581. return aMap ;
  582. }
  583. // This function is the inverse of _CreateTableMap - it takes in a table map and converts it to an HTML table.
  584. FCKTableHandler._InstallTableMap = function( tableMap, table )
  585. {
  586. // Workaround for #1917 : MSIE will always report a cell's rowSpan as 1 as long
  587. // as the cell is not attached to a row. So we'll need an alternative attribute
  588. // for storing the calculated rowSpan in IE.
  589. var rowSpanAttr = FCKBrowserInfo.IsIE ? "_fckrowspan" : "rowSpan" ;
  590. // Disconnect all the cells in tableMap from their parents, set all colSpan and rowSpan attributes to 1.
  591. for ( var i = 0 ; i < tableMap.length ; i++ )
  592. {
  593. for ( var j = 0 ; j < tableMap[i].length ; j++ )
  594. {
  595. var cell = tableMap[i][j] ;
  596. if ( cell.parentNode )
  597. cell.parentNode.removeChild( cell ) ;
  598. cell.colSpan = cell[rowSpanAttr] = 1 ;
  599. }
  600. }
  601. // Scan by rows and set colSpan.
  602. var maxCol = 0 ;
  603. for ( var i = 0 ; i < tableMap.length ; i++ )
  604. {
  605. for ( var j = 0 ; j < tableMap[i].length ; j++ )
  606. {
  607. var cell = tableMap[i][j] ;
  608. if ( ! cell)
  609. continue ;
  610. if ( j > maxCol )
  611. maxCol = j ;
  612. if ( cell._colScanned === true )
  613. continue ;
  614. if ( tableMap[i][j-1] == cell )
  615. cell.colSpan++ ;
  616. if ( tableMap[i][j+1] != cell )
  617. cell._colScanned = true ;
  618. }
  619. }
  620. // Scan by columns and set rowSpan.
  621. for ( var i = 0 ; i <= maxCol ; i++ )
  622. {
  623. for ( var j = 0 ; j < tableMap.length ; j++ )
  624. {
  625. if ( ! tableMap[j] )
  626. continue ;
  627. var cell = tableMap[j][i] ;
  628. if ( ! cell || cell._rowScanned === true )
  629. continue ;
  630. if ( tableMap[j-1] && tableMap[j-1][i] == cell )
  631. cell[rowSpanAttr]++ ;
  632. if ( ! tableMap[j+1] || tableMap[j+1][i] != cell )
  633. cell._rowScanned = true ;
  634. }
  635. }
  636. // Clear all temporary flags.
  637. for ( var i = 0 ; i < tableMap.length ; i++ )
  638. {
  639. for ( var j = 0 ; j < tableMap[i].length ; j++)
  640. {
  641. var cell = tableMap[i][j] ;
  642. FCKDomTools.ClearElementJSProperty(cell, '_colScanned' ) ;
  643. FCKDomTools.ClearElementJSProperty(cell, '_rowScanned' ) ;
  644. }
  645. }
  646. // Insert physical rows and columns to the table.
  647. for ( var i = 0 ; i < tableMap.length ; i++ )
  648. {
  649. var rowObj = FCK.EditorDocument.createElement( 'tr' ) ;
  650. for ( var j = 0 ; j < tableMap[i].length ; )
  651. {
  652. var cell = tableMap[i][j] ;
  653. if ( tableMap[i-1] && tableMap[i-1][j] == cell )
  654. {
  655. j += cell.colSpan ;
  656. continue ;
  657. }
  658. rowObj.appendChild( cell ) ;
  659. if ( rowSpanAttr != 'rowSpan' )
  660. {
  661. cell.rowSpan = cell[rowSpanAttr] ;
  662. cell.removeAttribute( rowSpanAttr ) ;
  663. }
  664. j += cell.colSpan ;
  665. if ( cell.colSpan == 1 )
  666. cell.removeAttribute( 'colspan' ) ;
  667. if ( cell.rowSpan == 1 )
  668. cell.removeAttribute( 'rowspan' ) ;
  669. }
  670. if ( FCKBrowserInfo.IsIE )
  671. {
  672. table.rows[i].replaceNode( rowObj ) ;
  673. }
  674. else
  675. {
  676. table.rows[i].innerHTML = '' ;
  677. FCKDomTools.MoveChildren( rowObj, table.rows[i] ) ;
  678. }
  679. }
  680. }
  681. FCKTableHandler._MoveCaretToCell = function ( refCell, toStart )
  682. {
  683. var range = new FCKDomRange( FCK.EditorWindow ) ;
  684. range.MoveToNodeContents( refCell ) ;
  685. range.Collapse( toStart ) ;
  686. range.Select() ;
  687. }
  688. FCKTableHandler.ClearRow = function( tr )
  689. {
  690. // Get the array of row's cells.
  691. var aCells = tr.cells ;
  692. // Replace the contents of each cell with "nothing".
  693. for ( var i = 0 ; i < aCells.length ; i++ )
  694. {
  695. aCells[i].innerHTML = '' ;
  696. if ( FCKBrowserInfo.IsGeckoLike )
  697. FCKTools.AppendBogusBr( aCells[i] ) ;
  698. }
  699. }
  700. FCKTableHandler.GetMergeRightTarget = function()
  701. {
  702. var cells = this.GetSelectedCells() ;
  703. if ( cells.length != 1 )
  704. return null ;
  705. var refCell = cells[0] ;
  706. var tableMap = this._CreateTableMap( refCell ) ;
  707. var rowIdx = refCell.parentNode.rowIndex ;
  708. var colIdx = this._GetCellIndexSpan( tableMap, rowIdx, refCell ) ;
  709. var nextColIdx = colIdx + ( isNaN( refCell.colSpan ) ? 1 : refCell.colSpan ) ;
  710. var nextCell = tableMap[rowIdx][nextColIdx] ;
  711. if ( ! nextCell )
  712. return null ;
  713. // The two cells must have the same vertical geometry, otherwise merging does not make sense.
  714. this._MarkCells( [refCell, nextCell], '_SizeTest' ) ;
  715. var refGeometry = this._GetMarkerGeometry( tableMap, rowIdx, colIdx, '_SizeTest' ) ;
  716. var nextGeometry = this._GetMarkerGeometry( tableMap, rowIdx, nextColIdx, '_SizeTest' ) ;
  717. this._UnmarkCells( [refCell, nextCell], '_SizeTest' ) ;
  718. if ( refGeometry.height != nextGeometry.height || refGeometry.y != nextGeometry.y )
  719. return null ;
  720. return { 'refCell' : refCell, 'nextCell' : nextCell, 'tableMap' : tableMap } ;
  721. }
  722. FCKTableHandler.GetMergeDownTarget = function()
  723. {
  724. var cells = this.GetSelectedCells() ;
  725. if ( cells.length != 1 )
  726. return null ;
  727. var refCell = cells[0] ;
  728. var tableMap = this._CreateTableMap( refCell ) ;
  729. var rowIdx = refCell.parentNode.rowIndex ;
  730. var colIdx = this._GetCellIndexSpan( tableMap, rowIdx, refCell ) ;
  731. var newRowIdx = rowIdx + ( isNaN( refCell.rowSpan ) ? 1 : refCell.rowSpan ) ;
  732. if ( ! tableMap[newRowIdx] )
  733. return null ;
  734. var nextCell = tableMap[newRowIdx][colIdx] ;
  735. if ( ! nextCell )
  736. return null ;
  737. // Check if the selected cells are both in the same table section (thead, tfoot or tbody).
  738. if ( refCell.parentNode.parentNode != nextCell.parentNode.parentNode )
  739. return null ;
  740. // The two cells must have the same horizontal geometry, otherwise merging does not makes sense.
  741. this._MarkCells( [refCell, nextCell], '_SizeTest' ) ;
  742. var refGeometry = this._GetMarkerGeometry( tableMap, rowIdx, colIdx, '_SizeTest' ) ;
  743. var nextGeometry = this._GetMarkerGeometry( tableMap, newRowIdx, colIdx, '_SizeTest' ) ;
  744. this._UnmarkCells( [refCell, nextCell], '_SizeTest' ) ;
  745. if ( refGeometry.width != nextGeometry.width || refGeometry.x != nextGeometry.x )
  746. return null ;
  747. return { 'refCell' : refCell, 'nextCell' : nextCell, 'tableMap' : tableMap } ;
  748. }