Fixed bug #14202: htmlArea RTE: Tables may get lost when using remove format feature
[Packages/TYPO3.CMS.git] / typo3 / wizard_table.php
1 <?php
2 /***************************************************************
3 * Copyright notice
4 *
5 * (c) 1999-2009 Kasper Skaarhoj (kasperYYYY@typo3.com)
6 * All rights reserved
7 *
8 * This script is part of the TYPO3 project. The TYPO3 project is
9 * free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * The GNU General Public License can be found at
15 * http://www.gnu.org/copyleft/gpl.html.
16 * A copy is found in the textfile GPL.txt and important notices to the license
17 * from the author is found in LICENSE.txt distributed with these scripts.
18 *
19 *
20 * This script is distributed in the hope that it will be useful,
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 * GNU General Public License for more details.
24 *
25 * This copyright notice MUST APPEAR in all copies of the script!
26 ***************************************************************/
27 /**
28 * Wizard to help make tables (eg. for tt_content elements) of type "table".
29 * Each line is a table row, each cell divided by a |
30 *
31 * $Id$
32 * Revised for TYPO3 3.6 November/2003 by Kasper Skaarhoj
33 * XHTML compliant
34 *
35 * @author Kasper Skaarhoj <kasperYYYY@typo3.com>
36 */
37 /**
38 * [CLASS/FUNCTION INDEX of SCRIPT]
39 *
40 *
41 *
42 * 84: class SC_wizard_table
43 * 116: function init()
44 * 158: function main()
45 * 173: function printContent()
46 * 184: function tableWizard()
47 *
48 * SECTION: Helper functions
49 * 223: function getConfigCode($row)
50 * 293: function getTableHTML($cfgArr,$row)
51 * 450: function changeFunc()
52 * 572: function cfgArray2CfgString($cfgArr)
53 * 603: function cfgString2CfgArray($cfgStr,$cols)
54 *
55 * TOTAL FUNCTIONS: 9
56 * (This index is automatically created/updated by the extension "extdeveval")
57 *
58 */
59
60
61
62 $BACK_PATH='';
63 require ('init.php');
64 require ('template.php');
65 $LANG->includeLLFile('EXT:lang/locallang_wizards.xml');
66
67
68
69
70
71
72
73
74
75
76
77 /**
78 * Script Class for rendering the Table Wizard
79 *
80 * @author Kasper Skaarhoj <kasperYYYY@typo3.com>
81 * @package TYPO3
82 * @subpackage core
83 */
84 class SC_wizard_table {
85
86 // Internal, dynamic:
87 /**
88 * document template object
89 *
90 * @var mediumDoc
91 */
92 var $doc;
93 var $content; // Content accumulation for the module.
94 var $include_once=array(); // List of files to include.
95 var $inputStyle=0; // True, then <input> fields are shown, not textareas.
96
97
98 // Internal, static:
99 var $xmlStorage=0; // If set, the string version of the content is interpreted/written as XML instead of the original linebased kind. This variable still needs binding to the wizard parameters - but support is ready!
100 var $numNewRows=1; // Number of new rows to add in bottom of wizard
101 var $colsFieldName='cols'; // Name of field in parent record which MAY contain the number of columns for the table - here hardcoded to the value of tt_content. Should be set by TCEform parameters (from P)
102
103
104 // Internal, static: GPvars
105 var $P; // Wizard parameters, coming from TCEforms linking to the wizard.
106 var $TABLECFG; // The array which is constantly submitted by the multidimensional form of this wizard.
107
108 // table parsing
109 var $tableParsing_quote; // quoting of table cells
110 var $tableParsing_delimiter; // delimiter between table cells
111
112
113
114
115
116 /**
117 * Initialization of the class
118 *
119 * @return void
120 */
121 function init() {
122 // GPvars:
123 $this->P = t3lib_div::_GP('P');
124 $this->TABLECFG = t3lib_div::_GP('TABLE');
125
126 // Setting options:
127 $this->xmlStorage = $this->P['params']['xmlOutput'];
128 $this->numNewRows = t3lib_div::intInRange($this->P['params']['numNewRows'],1,50,5);
129
130 // Textareas or input fields:
131 $this->inputStyle=isset($this->TABLECFG['textFields']) ? $this->TABLECFG['textFields'] : 1;
132
133 // Document template object:
134 $this->doc = t3lib_div::makeInstance('template');
135 $this->doc->backPath = $GLOBALS['BACK_PATH'];
136 $this->doc->setModuleTemplate('templates/wizard_table.html');
137 $this->doc->JScode=$this->doc->wrapScriptTags('
138 function jumpToUrl(URL,formEl) { //
139 window.location.href = URL;
140 }
141 ');
142
143 // Setting form tag:
144 list($rUri) = explode('#',t3lib_div::getIndpEnv('REQUEST_URI'));
145 $this->doc->form ='<form action="'.htmlspecialchars($rUri).'" method="post" name="wizardForm">';
146
147 // If save command found, include tcemain:
148 if ($_POST['savedok_x'] || $_POST['saveandclosedok_x']) {
149 $this->include_once[]=PATH_t3lib.'class.t3lib_tcemain.php';
150 }
151
152 $this->tableParsing_delimiter = '|';
153 $this->tableParsing_quote = '';
154 }
155
156 /**
157 * Main function, rendering the table wizard
158 *
159 * @return void
160 */
161 function main() {
162 if ($this->P['table'] && $this->P['field'] && $this->P['uid']) {
163 $this->content.= $this->doc->section($GLOBALS['LANG']->getLL('table_title'), $this->tableWizard(), 0, 1);
164 } else {
165 $this->content.= $this->doc->section($GLOBALS['LANG']->getLL('table_title'), '<span class="typo3-red">' . $GLOBALS['LANG']->getLL('table_noData',1) . '</span>', 0, 1);
166 }
167
168 // Setting up the buttons and markers for docheader
169 $docHeaderButtons = $this->getButtons();
170 $markers['CSH'] = $docHeaderButtons['csh'];
171 $markers['CONTENT'] = $this->content;
172
173 // Build the <body> for the module
174 $this->content = $this->doc->startPage('Table');
175 $this->content.= $this->doc->moduleBody($this->pageinfo, $docHeaderButtons, $markers);
176 $this->content.= $this->doc->endPage();
177 $this->content = $this->doc->insertStylesAndJS($this->content);
178 }
179
180 /**
181 * Outputting the accumulated content to screen
182 *
183 * @return void
184 */
185 function printContent() {
186 echo $this->content;
187 }
188
189 /**
190 * Create the panel of buttons for submitting the form or otherwise perform operations.
191 *
192 * @return array all available buttons as an assoc. array
193 */
194 protected function getButtons() {
195 $buttons = array(
196 'csh' => '',
197 'csh_buttons' => '',
198 'close' => '',
199 'save' => '',
200 'save_close' => '',
201 'reload' => '',
202 );
203
204 if ($this->P['table'] && $this->P['field'] && $this->P['uid']) {
205 // CSH
206 $buttons['csh'] = t3lib_BEfunc::cshItem('xMOD_csh_corebe', 'wizard_table_wiz', $GLOBALS['BACK_PATH'], '');
207
208 // CSH Buttons
209 $buttons['csh_buttons'] = t3lib_BEfunc::cshItem('xMOD_csh_corebe', 'wizard_table_wiz_buttons', $GLOBALS['BACK_PATH'], '');
210
211 // Close
212 $buttons['close'] = '<a href="#" onclick="' . htmlspecialchars('jumpToUrl(unescape(\'' . rawurlencode($this->P['returnUrl']) . '\')); return false;') . '">' .
213 '<img' . t3lib_iconWorks::skinImg($this->doc->backPath, 'gfx/closedok.gif') . ' class="c-inputButton" title="' . $GLOBALS['LANG']->sL('LLL:EXT:lang/locallang_core.php:rm.closeDoc', 1) . '" alt="" />' .
214 '</a>';
215
216 // Save
217 $buttons['save'] = '<input type="image" class="c-inputButton" name="savedok"' . t3lib_iconWorks::skinImg($this->doc->backPath, 'gfx/savedok.gif') . ' title="' . $GLOBALS['LANG']->sL('LLL:EXT:lang/locallang_core.php:rm.saveDoc', 1) . '" />';
218
219 // Save & Close
220 $buttons['save_close'] = '<input type="image" class="c-inputButton" name="saveandclosedok"' . t3lib_iconWorks::skinImg($this->doc->backPath, 'gfx/saveandclosedok.gif') . ' title="' . $GLOBALS['LANG']->sL('LLL:EXT:lang/locallang_core.php:rm.saveCloseDoc', 1) . '" />';
221
222 // Reload
223 $buttons['reload'] = '<input type="image" class="c-inputButton" name="_refresh"' . t3lib_iconWorks::skinImg($this->doc->backPath, 'gfx/refresh_n.gif') . ' title="' . $GLOBALS['LANG']->getLL('forms_refresh', 1) . '" />';
224 }
225
226 return $buttons;
227 }
228
229 /**
230 * Draws the table wizard content
231 *
232 * @return string HTML content for the form.
233 */
234 function tableWizard() {
235
236 // First, check the references by selecting the record:
237 $row = t3lib_BEfunc::getRecord($this->P['table'],$this->P['uid']);
238 if (!is_array($row)) {
239 t3lib_BEfunc::typo3PrintError ('Wizard Error','No reference to record',0);
240 exit;
241 }
242
243 // This will get the content of the form configuration code field to us - possibly cleaned up, saved to database etc. if the form has been submitted in the meantime.
244 $tableCfgArray = $this->getConfigCode($row);
245
246 // Generation of the Table Wizards HTML code:
247 $content = $this->getTableHTML($tableCfgArray,$row);
248
249 // Return content:
250 return $content;
251 }
252
253
254
255
256
257
258
259 /***************************
260 *
261 * Helper functions
262 *
263 ***************************/
264
265 /**
266 * Will get and return the configuration code string
267 * Will also save (and possibly redirect/exit) the content if a save button has been pressed
268 *
269 * @param array Current parent record row
270 * @return array Table config code in an array
271 * @access private
272 */
273 function getConfigCode($row) {
274
275 // get delimiter settings
276 $flexForm = t3lib_div::xml2array($row['pi_flexform']);
277
278 if (is_array($flexForm)) {
279 $this->tableParsing_quote = $flexForm['data']['s_parsing']['lDEF']['tableparsing_quote']['vDEF']?chr(intval($flexForm['data']['s_parsing']['lDEF']['tableparsing_quote']['vDEF'])):'';
280 $this->tableParsing_delimiter = $flexForm['data']['s_parsing']['lDEF']['tableparsing_delimiter']['vDEF']?chr(intval($flexForm['data']['s_parsing']['lDEF']['tableparsing_delimiter']['vDEF'])):'|';
281 }
282
283 // If some data has been submitted, then construct
284 if (isset($this->TABLECFG['c'])) {
285
286 // Process incoming:
287 $this->changeFunc();
288
289
290 // Convert to string (either line based or XML):
291 if ($this->xmlStorage) {
292 // Convert the input array to XML:
293 $bodyText = t3lib_div::array2xml_cs($this->TABLECFG['c'],'T3TableWizard');
294
295 // Setting cfgArr directly from the input:
296 $cfgArr = $this->TABLECFG['c'];
297 } else {
298 // Convert the input array to a string of configuration code:
299 $bodyText = $this->cfgArray2CfgString($this->TABLECFG['c']);
300
301 // Create cfgArr from the string based configuration - that way it is cleaned up and any incompatibilities will be removed!
302 $cfgArr = $this->cfgString2CfgArray($bodyText,$row[$this->colsFieldName]);
303 }
304
305 // If a save button has been pressed, then save the new field content:
306 if ($_POST['savedok_x'] || $_POST['saveandclosedok_x']) {
307
308 // Make TCEmain object:
309 $tce = t3lib_div::makeInstance('t3lib_TCEmain');
310 $tce->stripslashes_values=0;
311
312 // Put content into the data array:
313 $data=array();
314 $data[$this->P['table']][$this->P['uid']][$this->P['field']]=$bodyText;
315
316 // Perform the update:
317 $tce->start($data,array());
318 $tce->process_datamap();
319
320 // If the save/close button was pressed, then redirect the screen:
321 if ($_POST['saveandclosedok_x']) {
322 t3lib_utility_Http::redirect($this->P['returnUrl']);
323 }
324 }
325 } else { // If nothing has been submitted, load the $bodyText variable from the selected database row:
326 if ($this->xmlStorage) {
327 $cfgArr = t3lib_div::xml2array($row[$this->P['field']]);
328 } else { // Regular linebased table configuration:
329 $cfgArr = $this->cfgString2CfgArray($row[$this->P['field']],$row[$this->colsFieldName]);
330 }
331 $cfgArr = is_array($cfgArr) ? $cfgArr : array();
332 }
333
334 return $cfgArr;
335 }
336
337 /**
338 * Creates the HTML for the Table Wizard:
339 *
340 * @param array Table config array
341 * @param array Current parent record array
342 * @return string HTML for the table wizard
343 * @access private
344 */
345 function getTableHTML($cfgArr,$row) {
346 global $LANG;
347
348 // Traverse the rows:
349 $tRows=array();
350 $k=0;
351 foreach($cfgArr as $cellArr) {
352 if (is_array($cellArr)) {
353 // Initialize:
354 $cells=array();
355 $a=0;
356
357 // Traverse the columns:
358 foreach($cellArr as $cellContent) {
359 if ($this->inputStyle) {
360 $cells[]='<input type="text"'.$this->doc->formWidth(20).' name="TABLE[c]['.(($k+1)*2).']['.(($a+1)*2).']" value="'.htmlspecialchars($cellContent).'" />';
361 } else {
362 $cellContent=preg_replace('/<br[ ]?[\/]?>/i',LF,$cellContent);
363 $cells[]='<textarea '.$this->doc->formWidth(20).' rows="5" name="TABLE[c]['.(($k+1)*2).']['.(($a+1)*2).']">'.t3lib_div::formatForTextarea($cellContent).'</textarea>';
364 }
365
366 // Increment counter:
367 $a++;
368 }
369
370 // CTRL panel for a table row (move up/down/around):
371 $onClick="document.wizardForm.action+='#ANC_".(($k+1)*2-2)."';";
372 $onClick=' onclick="'.htmlspecialchars($onClick).'"';
373 $ctrl='';
374
375 $brTag=$this->inputStyle?'':'<br />';
376 if ($k!=0) {
377 $ctrl.='<input type="image" name="TABLE[row_up]['.(($k+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/pil2up.gif','').$onClick.' title="'.$LANG->getLL('table_up',1).'" />'.$brTag;
378 } else {
379 $ctrl.='<input type="image" name="TABLE[row_bottom]['.(($k+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/turn_up.gif','').$onClick.' title="'.$LANG->getLL('table_bottom',1).'" />'.$brTag;
380 }
381 $ctrl.='<input type="image" name="TABLE[row_remove]['.(($k+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/garbage.gif','').$onClick.' title="'.$LANG->getLL('table_removeRow',1).'" />'.$brTag;
382
383 // FIXME what is $tLines? See wizard_forms.php for the same.
384 if (($k+1)!=count($tLines)) {
385 $ctrl.='<input type="image" name="TABLE[row_down]['.(($k+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/pil2down.gif','').$onClick.' title="'.$LANG->getLL('table_down',1).'" />'.$brTag;
386 } else {
387 $ctrl.='<input type="image" name="TABLE[row_top]['.(($k+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/turn_down.gif','').$onClick.' title="'.$LANG->getLL('table_top',1).'" />'.$brTag;
388 }
389 $ctrl.='<input type="image" name="TABLE[row_add]['.(($k+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/add.gif','').$onClick.' title="'.$LANG->getLL('table_addRow',1).'" />'.$brTag;
390
391 $tRows[]='
392 <tr class="bgColor4">
393 <td class="bgColor5"><a name="ANC_'.(($k+1)*2).'"></a><span class="c-wizButtonsV">'.$ctrl.'</span></td>
394 <td>'.implode('</td>
395 <td>',$cells).'</td>
396 </tr>';
397
398 // Increment counter:
399 $k++;
400 }
401 }
402
403 // CTRL panel for a table column (move left/right/around/delete)
404 $cells=array();
405 $cells[]='';
406 // Finding first row:
407 reset($cfgArr);
408 $firstRow=current($cfgArr);
409 if (is_array($firstRow)) {
410
411 // Init:
412 $a=0;
413 $cols=count($firstRow);
414
415 // Traverse first row:
416 foreach($firstRow as $temp) {
417 $ctrl='';
418 if ($a!=0) {
419 $ctrl.='<input type="image" name="TABLE[col_left]['.(($a+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/pil2left.gif','').' title="'.$LANG->getLL('table_left',1).'" />';
420 } else {
421 $ctrl.='<input type="image" name="TABLE[col_end]['.(($a+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/turn_left.gif','').' title="'.$LANG->getLL('table_end',1).'" />';
422 }
423 $ctrl.='<input type="image" name="TABLE[col_remove]['.(($a+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/garbage.gif','').' title="'.$LANG->getLL('table_removeColumn',1).'" />';
424 if (($a+1)!=$cols) {
425 $ctrl.='<input type="image" name="TABLE[col_right]['.(($a+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/pil2right.gif','').' title="'.$LANG->getLL('table_right',1).'" />';
426 } else {
427 $ctrl.='<input type="image" name="TABLE[col_start]['.(($a+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/turn_right.gif','').' title="'.$LANG->getLL('table_start',1).'" />';
428 }
429 $ctrl.='<input type="image" name="TABLE[col_add]['.(($a+1)*2).']"'.t3lib_iconWorks::skinImg($this->doc->backPath,'gfx/add.gif','').' title="'.$LANG->getLL('table_addColumn',1).'" />';
430 $cells[]='<span class="c-wizButtonsH">'.$ctrl.'</span>';
431
432 // Incr. counter:
433 $a++;
434 }
435 $tRows[]='
436 <tr class="bgColor5">
437 <td align="center">'.implode('</td>
438 <td align="center">',$cells).'</td>
439 </tr>';
440 }
441
442 $content = '';
443
444 // Implode all table rows into a string, wrapped in table tags.
445 $content.= '
446
447
448 <!--
449 Table wizard
450 -->
451 <table border="0" cellpadding="0" cellspacing="1" id="typo3-tablewizard">
452 '.implode('',$tRows).'
453 </table>';
454
455 // Input type checkbox:
456 $content.= '
457
458 <!--
459 Input mode check box:
460 -->
461 <div id="c-inputMode">
462 '.
463 '<input type="hidden" name="TABLE[textFields]" value="0" />'.
464 '<input type="checkbox" name="TABLE[textFields]" id="textFields" value="1"'.($this->inputStyle?' checked="checked"':'').' /> <label for="textFields">'.
465 $LANG->getLL('table_smallFields').'</label>
466 </div>
467
468 <br /><br />
469 ';
470
471 // Return content:
472 return $content;
473 }
474
475 /**
476 * Detects if a control button (up/down/around/delete) has been pressed for an item and accordingly it will manipulate the internal TABLECFG array
477 *
478 * @return void
479 * @access private
480 */
481 function changeFunc() {
482 if ($this->TABLECFG['col_remove']) {
483 $kk = key($this->TABLECFG['col_remove']);
484 $cmd='col_remove';
485 } elseif ($this->TABLECFG['col_add']) {
486 $kk = key($this->TABLECFG['col_add']);
487 $cmd='col_add';
488 } elseif ($this->TABLECFG['col_start']) {
489 $kk = key($this->TABLECFG['col_start']);
490 $cmd='col_start';
491 } elseif ($this->TABLECFG['col_end']) {
492 $kk = key($this->TABLECFG['col_end']);
493 $cmd='col_end';
494 } elseif ($this->TABLECFG['col_left']) {
495 $kk = key($this->TABLECFG['col_left']);
496 $cmd='col_left';
497 } elseif ($this->TABLECFG['col_right']) {
498 $kk = key($this->TABLECFG['col_right']);
499 $cmd='col_right';
500 } elseif ($this->TABLECFG['row_remove']) {
501 $kk = key($this->TABLECFG['row_remove']);
502 $cmd='row_remove';
503 } elseif ($this->TABLECFG['row_add']) {
504 $kk = key($this->TABLECFG['row_add']);
505 $cmd='row_add';
506 } elseif ($this->TABLECFG['row_top']) {
507 $kk = key($this->TABLECFG['row_top']);
508 $cmd='row_top';
509 } elseif ($this->TABLECFG['row_bottom']) {
510 $kk = key($this->TABLECFG['row_bottom']);
511 $cmd='row_bottom';
512 } elseif ($this->TABLECFG['row_up']) {
513 $kk = key($this->TABLECFG['row_up']);
514 $cmd='row_up';
515 } elseif ($this->TABLECFG['row_down']) {
516 $kk = key($this->TABLECFG['row_down']);
517 $cmd='row_down';
518 }
519
520 if ($cmd && t3lib_div::testInt($kk)) {
521 if (substr($cmd,0,4)=='row_') {
522 switch($cmd) {
523 case 'row_remove':
524 unset($this->TABLECFG['c'][$kk]);
525 break;
526 case 'row_add':
527 for($a=1;$a<=$this->numNewRows;$a++) {
528 if (!isset($this->TABLECFG['c'][$kk+$a])) { // Checking if set: The point is that any new row inbetween existing rows will be true after one row is added while if rows are added in the bottom of the table there will be no existing rows to stop the addition of new rows which means it will add up to $this->numNewRows rows then.
529 $this->TABLECFG['c'][$kk+$a] = array();
530 } else {
531 break;
532 }
533 }
534 break;
535 case 'row_top':
536 $this->TABLECFG['c'][1]=$this->TABLECFG['c'][$kk];
537 unset($this->TABLECFG['c'][$kk]);
538 break;
539 case 'row_bottom':
540 $this->TABLECFG['c'][10000000]=$this->TABLECFG['c'][$kk];
541 unset($this->TABLECFG['c'][$kk]);
542 break;
543 case 'row_up':
544 $this->TABLECFG['c'][$kk-3]=$this->TABLECFG['c'][$kk];
545 unset($this->TABLECFG['c'][$kk]);
546 break;
547 case 'row_down':
548 $this->TABLECFG['c'][$kk+3]=$this->TABLECFG['c'][$kk];
549 unset($this->TABLECFG['c'][$kk]);
550 break;
551 }
552 ksort($this->TABLECFG['c']);
553 }
554 if (substr($cmd,0,4)=='col_') {
555 reset($this->TABLECFG['c']);
556 while(list($cAK)=each($this->TABLECFG['c'])) {
557 switch($cmd) {
558 case 'col_remove':
559 unset($this->TABLECFG['c'][$cAK][$kk]);
560 break;
561 case 'col_add':
562 $this->TABLECFG['c'][$cAK][$kk+1]='';
563 break;
564 case 'col_start':
565 $this->TABLECFG['c'][$cAK][1]=$this->TABLECFG['c'][$cAK][$kk];
566 unset($this->TABLECFG['c'][$cAK][$kk]);
567 break;
568 case 'col_end':
569 $this->TABLECFG['c'][$cAK][1000000]=$this->TABLECFG['c'][$cAK][$kk];
570 unset($this->TABLECFG['c'][$cAK][$kk]);
571 break;
572 case 'col_left':
573 $this->TABLECFG['c'][$cAK][$kk-3]=$this->TABLECFG['c'][$cAK][$kk];
574 unset($this->TABLECFG['c'][$cAK][$kk]);
575 break;
576 case 'col_right':
577 $this->TABLECFG['c'][$cAK][$kk+3]=$this->TABLECFG['c'][$cAK][$kk];
578 unset($this->TABLECFG['c'][$cAK][$kk]);
579 break;
580 }
581 ksort($this->TABLECFG['c'][$cAK]);
582 }
583 }
584 }
585
586 // Convert line breaks to <br /> tags:
587 reset($this->TABLECFG['c']);
588 while(list($a)=each($this->TABLECFG['c'])) {
589 reset($this->TABLECFG['c'][$a]);
590 while(list($b)=each($this->TABLECFG['c'][$a])) {
591 $this->TABLECFG['c'][$a][$b] = str_replace(LF,'<br />',str_replace(CR,'',$this->TABLECFG['c'][$a][$b]));
592 }
593 }
594 }
595
596 /**
597 * Converts the input array to a configuration code string
598 *
599 * @param array Array of table configuration (follows the input structure from the table wizard POST form)
600 * @return string The array converted into a string with line-based configuration.
601 * @see cfgString2CfgArray()
602 */
603 function cfgArray2CfgString($cfgArr) {
604
605 // Initialize:
606 $inLines=array();
607
608 // Traverse the elements of the table wizard and transform the settings into configuration code.
609 reset($this->TABLECFG['c']);
610 while(list($a)=each($this->TABLECFG['c'])) {
611 $thisLine=array();
612 reset($this->TABLECFG['c'][$a]);
613 while(list($b)=each($this->TABLECFG['c'][$a])) {
614 $thisLine[]=$this->tableParsing_quote.str_replace($this->tableParsing_delimiter,'',$this->TABLECFG['c'][$a][$b]).$this->tableParsing_quote;
615 }
616 $inLines[]=implode($this->tableParsing_delimiter,$thisLine);
617 }
618
619 // Finally, implode the lines into a string:
620 $bodyText = implode(LF,$inLines);
621
622 // Return the configuration code:
623 return $bodyText;
624 }
625
626 /**
627 * Converts the input configuration code string into an array
628 *
629 * @param string Configuration code
630 * @param integer Default number of columns
631 * @return array Configuration array
632 * @see cfgArray2CfgString()
633 */
634 function cfgString2CfgArray($cfgStr,$cols) {
635
636 // Explode lines in the configuration code - each line is a table row.
637 $tLines=explode(LF,$cfgStr);
638
639 // Setting number of columns
640 if (!$cols && trim($tLines[0])) { // auto...
641 $cols = count(explode($this->tableParsing_delimiter,$tLines[0]));
642 }
643 $cols=$cols?$cols:4;
644
645 // Traverse the number of table elements:
646 $cfgArr=array();
647 foreach($tLines as $k => $v) {
648
649 // Initialize:
650 $vParts = explode($this->tableParsing_delimiter,$v);
651
652 // Traverse columns:
653 for ($a=0;$a<$cols;$a++) {
654 if ($this->tableParsing_quote && substr($vParts[$a],0,1) == $this->tableParsing_quote && substr($vParts[$a],-1,1) == $this->tableParsing_quote) {
655 $vParts[$a] = substr(trim($vParts[$a]),1,-1);
656 }
657 $cfgArr[$k][$a]=$vParts[$a];
658 }
659 }
660
661 // Return configuration array:
662 return $cfgArr;
663 }
664 }
665
666
667 if (defined('TYPO3_MODE') && $TYPO3_CONF_VARS[TYPO3_MODE]['XCLASS']['typo3/wizard_table.php']) {
668 include_once($TYPO3_CONF_VARS[TYPO3_MODE]['XCLASS']['typo3/wizard_table.php']);
669 }
670
671
672
673 // Make instance:
674 $SOBE = t3lib_div::makeInstance('SC_wizard_table');
675 $SOBE->init();
676
677 // Include files?
678 foreach($SOBE->include_once as $INC_FILE) include_once($INC_FILE);
679
680 $SOBE->main();
681 $SOBE->printContent();
682
683 ?>