1) Added missing strings for italic, underline and strike through.
[phpeclipse.git] / net.sourceforge.phpeclipse / src / net / sourceforge / phpdt / internal / ui / preferences / CodeFormatterPreferencePage.java
1 /*\r
2  * (c) Copyright IBM Corp. 2000, 2001.\r
3  * All Rights Reserved.\r
4  */\r
5 package net.sourceforge.phpdt.internal.ui.preferences;\r
6 \r
7 import java.io.BufferedReader;\r
8 import java.io.IOException;\r
9 import java.io.InputStreamReader;\r
10 import java.util.ArrayList;\r
11 import java.util.Hashtable;\r
12 \r
13 import net.sourceforge.phpdt.core.ICodeFormatter;\r
14 import net.sourceforge.phpdt.core.JavaCore;\r
15 import net.sourceforge.phpdt.core.ToolFactory;\r
16 import net.sourceforge.phpdt.internal.ui.PHPUIMessages;\r
17 import net.sourceforge.phpdt.internal.ui.dialogs.StatusInfo;\r
18 import net.sourceforge.phpdt.internal.ui.dialogs.StatusUtil;\r
19 import net.sourceforge.phpdt.internal.ui.util.TabFolderLayout;\r
20 import net.sourceforge.phpeclipse.PHPeclipsePlugin;\r
21 \r
22 import org.eclipse.core.runtime.IStatus;\r
23 import org.eclipse.jface.preference.PreferencePage;\r
24 import org.eclipse.jface.text.Document;\r
25 import org.eclipse.jface.text.IDocument;\r
26 import org.eclipse.swt.SWT;\r
27 import org.eclipse.swt.events.ModifyEvent;\r
28 import org.eclipse.swt.events.ModifyListener;\r
29 import org.eclipse.swt.events.SelectionEvent;\r
30 import org.eclipse.swt.events.SelectionListener;\r
31 import org.eclipse.swt.layout.GridData;\r
32 import org.eclipse.swt.layout.GridLayout;\r
33 import org.eclipse.swt.widgets.Button;\r
34 import org.eclipse.swt.widgets.Composite;\r
35 import org.eclipse.swt.widgets.Control;\r
36 import org.eclipse.swt.widgets.Label;\r
37 import org.eclipse.swt.widgets.TabFolder;\r
38 import org.eclipse.swt.widgets.TabItem;\r
39 import org.eclipse.swt.widgets.Text;\r
40 import org.eclipse.ui.IWorkbench;\r
41 import org.eclipse.ui.IWorkbenchPreferencePage;\r
42 \r
43 /*\r
44  * The page for setting code formatter options\r
45  */\r
46 public class CodeFormatterPreferencePage extends PreferencePage implements\r
47         IWorkbenchPreferencePage {\r
48 \r
49     // Preference store keys, see PHPCore.getOptions\r
50     private static final String PREF_NEWLINE_OPENING_BRACES = JavaCore.FORMATTER_NEWLINE_OPENING_BRACE;\r
51 \r
52     private static final String PREF_NEWLINE_CONTROL_STATEMENT = JavaCore.FORMATTER_NEWLINE_CONTROL;\r
53 \r
54     private static final String PREF_NEWLINE_CLEAR_ALL = JavaCore.FORMATTER_CLEAR_BLANK_LINES;\r
55 \r
56     // private static final String PREF_NEWLINE_ELSE_IF=\r
57     // PHPCore.FORMATTER_NEWLINE_ELSE_IF;\r
58     private static final String PREF_NEWLINE_EMPTY_BLOCK = JavaCore.FORMATTER_NEWLINE_EMPTY_BLOCK;\r
59 \r
60     private static final String PREF_LINE_SPLIT = JavaCore.FORMATTER_LINE_SPLIT;\r
61 \r
62     private static final String PREF_STYLE_COMPACT_ASSIGNEMENT = JavaCore.FORMATTER_COMPACT_ASSIGNMENT;\r
63 \r
64     private static final String PREF_STYLE_COMPACT_STRING_CONCATENATION = JavaCore.FORMATTER_COMPACT_STRING_CONCATENATION;\r
65 \r
66     private static final String PREF_STYLE_COMPACT_ARRAYS = JavaCore.FORMATTER_COMPACT_ARRAYS;\r
67     \r
68     private static final String PREF_TAB_CHAR = JavaCore.FORMATTER_TAB_CHAR;\r
69 \r
70     private static final String PREF_TAB_SIZE = JavaCore.FORMATTER_TAB_SIZE;\r
71 \r
72     // values\r
73     private static final String INSERT = JavaCore.INSERT;\r
74 \r
75     private static final String DO_NOT_INSERT = JavaCore.DO_NOT_INSERT;\r
76 \r
77     private static final String COMPACT = JavaCore.COMPACT;\r
78 \r
79     private static final String NORMAL = JavaCore.NORMAL;\r
80 \r
81     private static final String TAB = JavaCore.TAB;\r
82 \r
83     private static final String SPACE = JavaCore.SPACE;\r
84 \r
85     private static final String CLEAR_ALL = JavaCore.CLEAR_ALL;\r
86 \r
87     private static final String PRESERVE_ONE = JavaCore.PRESERVE_ONE;\r
88 \r
89     private static String[] getAllKeys() {\r
90         return new String[] { PREF_NEWLINE_OPENING_BRACES,\r
91                 PREF_NEWLINE_CONTROL_STATEMENT, PREF_NEWLINE_CLEAR_ALL,\r
92                 // PREF_NEWLINE_ELSE_IF,\r
93                 PREF_NEWLINE_EMPTY_BLOCK, PREF_LINE_SPLIT,\r
94                 PREF_STYLE_COMPACT_ASSIGNEMENT, PREF_STYLE_COMPACT_STRING_CONCATENATION,\r
95                 PREF_STYLE_COMPACT_ARRAYS,\r
96                 PREF_TAB_CHAR, PREF_TAB_SIZE };\r
97     }\r
98 \r
99     /**\r
100      * Gets the currently configured tab size\r
101      * \r
102      * @deprecated Inline to avoid reference to preference page\r
103      */\r
104     public static int getTabSize() {\r
105         String string = (String) JavaCore.getOptions().get(PREF_TAB_SIZE);\r
106         return getPositiveIntValue(string, 4);\r
107     }\r
108 \r
109     /**\r
110      * Gets the current compating assignement configuration\r
111      * \r
112      * @deprecated Inline to avoid reference to preference page\r
113      */\r
114     public static boolean isCompactingAssignment() {\r
115         return COMPACT.equals(JavaCore.getOptions().get(\r
116                 PREF_STYLE_COMPACT_ASSIGNEMENT));\r
117     }\r
118 \r
119     /**\r
120      * Gets the current compating assignement configuration\r
121      * \r
122      * @deprecated Inline to avoid reference to preference page\r
123      */\r
124     public static boolean useSpaces() {\r
125         return SPACE.equals(JavaCore.getOptions().get(PREF_TAB_CHAR));\r
126     }\r
127 \r
128     private static int getPositiveIntValue(String string, int dflt) {\r
129         try {\r
130             int i = Integer.parseInt(string);\r
131             if (i >= 0) {\r
132                 return i;\r
133             }\r
134         } catch (NumberFormatException e) {\r
135         }\r
136         return dflt;\r
137     }\r
138 \r
139     private static class ControlData {\r
140         private String fKey;\r
141 \r
142         private String[] fValues;\r
143 \r
144         public ControlData(String key, String[] values) {\r
145             fKey = key;\r
146             fValues = values;\r
147         }\r
148 \r
149         public String getKey() {\r
150             return fKey;\r
151         }\r
152 \r
153         public String getValue(boolean selection) {\r
154             int index = selection ? 0 : 1;\r
155             return fValues[index];\r
156         }\r
157 \r
158         public String getValue(int index) {\r
159             return fValues[index];\r
160         }\r
161 \r
162         public int getSelection(String value) {\r
163             for (int i = 0; i < fValues.length; i++) {\r
164                 if (value.equals(fValues[i])) {\r
165                     return i;\r
166                 }\r
167             }\r
168             throw new IllegalArgumentException();\r
169         }\r
170     }\r
171 \r
172     private Hashtable fWorkingValues;\r
173 \r
174     private ArrayList fCheckBoxes;\r
175 \r
176     private ArrayList fTextBoxes;\r
177 \r
178     private SelectionListener fButtonSelectionListener;\r
179 \r
180     private ModifyListener fTextModifyListener;\r
181 \r
182     private String fPreviewText;\r
183 \r
184     private IDocument fPreviewDocument;\r
185 \r
186     private Text fTabSizeTextBox;\r
187 \r
188     // private SourceViewer fSourceViewer;\r
189 \r
190     public CodeFormatterPreferencePage() {\r
191         setPreferenceStore(PHPeclipsePlugin.getDefault().getPreferenceStore());\r
192         setDescription(PHPUIMessages\r
193                 .getString("CodeFormatterPreferencePage.description")); //$NON-NLS-1$\r
194 \r
195         fWorkingValues = JavaCore.getOptions();\r
196         fCheckBoxes = new ArrayList();\r
197         fTextBoxes = new ArrayList();\r
198 \r
199         fButtonSelectionListener = new SelectionListener() {\r
200             public void widgetDefaultSelected(SelectionEvent e) {\r
201             }\r
202 \r
203             public void widgetSelected(SelectionEvent e) {\r
204                 if (!e.widget.isDisposed()) {\r
205                     controlChanged((Button) e.widget);\r
206                 }\r
207             }\r
208         };\r
209 \r
210         fTextModifyListener = new ModifyListener() {\r
211             public void modifyText(ModifyEvent e) {\r
212                 if (!e.widget.isDisposed()) {\r
213                     textChanged((Text) e.widget);\r
214                 }\r
215             }\r
216         };\r
217 \r
218         fPreviewDocument = new Document();\r
219         fPreviewText = loadPreviewFile("CodeFormatterPreviewCode.txt"); //$NON-NLS-1$   \r
220     }\r
221 \r
222     /*\r
223      * @see IWorkbenchPreferencePage#init()\r
224      */\r
225     public void init(IWorkbench workbench) {\r
226     }\r
227 \r
228     /*\r
229      * @see PreferencePage#createControl(Composite)\r
230      */\r
231     public void createControl(Composite parent) {\r
232         super.createControl(parent);\r
233         // WorkbenchHelp.setHelp(getControl(),\r
234         // IJavaHelpContextIds.CODEFORMATTER_PREFERENCE_PAGE);\r
235     }\r
236 \r
237     /*\r
238      * @see PreferencePage#createContents(Composite)\r
239      */\r
240     protected Control createContents(Composite parent) {\r
241 \r
242         GridLayout layout = new GridLayout();\r
243         layout.marginHeight = 0;\r
244         layout.marginWidth = 0;\r
245 \r
246         Composite composite = new Composite(parent, SWT.NONE);\r
247         composite.setLayout(layout);\r
248 \r
249         TabFolder folder = new TabFolder(composite, SWT.NONE);\r
250         folder.setLayout(new TabFolderLayout());\r
251         folder.setLayoutData(new GridData(GridData.FILL_BOTH));\r
252 \r
253         String[] insertNotInsert = new String[] { INSERT, DO_NOT_INSERT };\r
254 \r
255         layout = new GridLayout();\r
256         layout.numColumns = 2;\r
257 \r
258         Composite newlineComposite = new Composite(folder, SWT.NULL);\r
259         newlineComposite.setLayout(layout);\r
260 \r
261         String label = PHPUIMessages\r
262                 .getString("CodeFormatterPreferencePage.newline_opening_braces.label"); //$NON-NLS-1$\r
263         addCheckBox(newlineComposite, label, PREF_NEWLINE_OPENING_BRACES,\r
264                 insertNotInsert);\r
265 \r
266         label = PHPUIMessages\r
267                 .getString("CodeFormatterPreferencePage.newline_control_statement.label"); //$NON-NLS-1$\r
268         addCheckBox(newlineComposite, label, PREF_NEWLINE_CONTROL_STATEMENT,\r
269                 insertNotInsert);\r
270 \r
271         label = PHPUIMessages\r
272                 .getString("CodeFormatterPreferencePage.newline_clear_lines"); //$NON-NLS-1$\r
273         addCheckBox(newlineComposite, label, PREF_NEWLINE_CLEAR_ALL,\r
274                 new String[] { CLEAR_ALL, PRESERVE_ONE });\r
275 \r
276         // label=\r
277         // PHPUIMessages.getString("CodeFormatterPreferencePage.newline_else_if.label");\r
278         // //$NON-NLS-1$\r
279         // addCheckBox(newlineComposite, label, PREF_NEWLINE_ELSE_IF,\r
280         // insertNotInsert);\r
281 \r
282         label = PHPUIMessages\r
283                 .getString("CodeFormatterPreferencePage.newline_empty_block.label"); //$NON-NLS-1$\r
284         addCheckBox(newlineComposite, label, PREF_NEWLINE_EMPTY_BLOCK,\r
285                 insertNotInsert);\r
286 \r
287         layout = new GridLayout();\r
288         layout.numColumns = 2;\r
289 \r
290         Composite lineSplittingComposite = new Composite(folder, SWT.NULL);\r
291         lineSplittingComposite.setLayout(layout);\r
292 \r
293         label = PHPUIMessages\r
294                 .getString("CodeFormatterPreferencePage.split_line.label"); //$NON-NLS-1$\r
295         addTextField(lineSplittingComposite, label, PREF_LINE_SPLIT);\r
296 \r
297         layout = new GridLayout();\r
298         layout.numColumns = 2;\r
299 \r
300         Composite styleComposite = new Composite(folder, SWT.NULL);\r
301         styleComposite.setLayout(layout);\r
302 \r
303         label = PHPUIMessages\r
304                 .getString("CodeFormatterPreferencePage.style_compact_assignement.label"); //$NON-NLS-1$\r
305         addCheckBox(styleComposite, label, PREF_STYLE_COMPACT_ASSIGNEMENT,\r
306                 new String[] { COMPACT, NORMAL });\r
307         \r
308         label = PHPUIMessages\r
309         .getString("CodeFormatterPreferencePage.style_compact_string_concatenation.label"); //$NON-NLS-1$\r
310         addCheckBox(styleComposite, label, PREF_STYLE_COMPACT_STRING_CONCATENATION,\r
311         new String[] { COMPACT, NORMAL });\r
312         \r
313         label = PHPUIMessages\r
314         .getString("CodeFormatterPreferencePage.style_compact_arrays.label"); //$NON-NLS-1$\r
315         addCheckBox(styleComposite, label, PREF_STYLE_COMPACT_ARRAYS,\r
316         new String[] { COMPACT, NORMAL });      \r
317 \r
318         label = PHPUIMessages\r
319                 .getString("CodeFormatterPreferencePage.tab_char.label"); //$NON-NLS-1$\r
320         addCheckBox(styleComposite, label, PREF_TAB_CHAR, new String[] { TAB,\r
321                 SPACE });\r
322 \r
323         label = PHPUIMessages\r
324                 .getString("CodeFormatterPreferencePage.tab_size.label"); //$NON-NLS-1$\r
325         fTabSizeTextBox = addTextField(styleComposite, label, PREF_TAB_SIZE);\r
326 \r
327         TabItem item = new TabItem(folder, SWT.NONE);\r
328         item.setText(PHPUIMessages\r
329                 .getString("CodeFormatterPreferencePage.tab.newline.tabtitle")); //$NON-NLS-1$\r
330         item.setControl(newlineComposite);\r
331 \r
332         item = new TabItem(folder, SWT.NONE);\r
333         item\r
334                 .setText(PHPUIMessages\r
335                         .getString("CodeFormatterPreferencePage.tab.linesplit.tabtitle")); //$NON-NLS-1$\r
336         item.setControl(lineSplittingComposite);\r
337 \r
338         item = new TabItem(folder, SWT.NONE);\r
339         item.setText(PHPUIMessages\r
340                 .getString("CodeFormatterPreferencePage.tab.style.tabtitle")); //$NON-NLS-1$\r
341         item.setControl(styleComposite);\r
342 \r
343         // fSourceViewer= createPreview(parent);\r
344 \r
345         updatePreview();\r
346 \r
347         return composite;\r
348     }\r
349 \r
350     // private SourceViewer createPreview(Composite parent) {\r
351     // SourceViewer previewViewer= new SourceViewer(parent, null, SWT.V_SCROLL |\r
352     // SWT.H_SCROLL | SWT.BORDER);\r
353     // JavaTextTools tools= JavaPlugin.getDefault().getJavaTextTools();\r
354     // previewViewer.configure(new PHPSourceViewerConfiguration(tools, null));\r
355     // previewViewer.getTextWidget().setFont(JFaceResources.getFontRegistry().get(JFaceResources.TEXT_FONT));\r
356     // previewViewer.getTextWidget().setTabs(getPositiveIntValue((String)\r
357     // fWorkingValues.get(PREF_TAB_SIZE), 0));\r
358     // previewViewer.setEditable(false);\r
359     // previewViewer.setDocument(fPreviewDocument);\r
360     // Control control= previewViewer.getControl();\r
361     // GridData gdata= new GridData(GridData.FILL_BOTH);\r
362     // gdata.widthHint= convertWidthInCharsToPixels(30);\r
363     // gdata.heightHint= convertHeightInCharsToPixels(5);\r
364     // control.setLayoutData(gdata);\r
365     // return previewViewer;\r
366     // }\r
367 \r
368     private Button addCheckBox(Composite parent, String label, String key,\r
369             String[] values) {\r
370         ControlData data = new ControlData(key, values);\r
371 \r
372         GridData gd = new GridData(GridData.FILL_HORIZONTAL);\r
373         gd.horizontalSpan = 2;\r
374 \r
375         Button checkBox = new Button(parent, SWT.CHECK);\r
376         checkBox.setText(label);\r
377         checkBox.setData(data);\r
378         checkBox.setLayoutData(gd);\r
379 \r
380         String currValue = (String) fWorkingValues.get(key);\r
381         checkBox.setSelection(data.getSelection(currValue) == 0);\r
382         checkBox.addSelectionListener(fButtonSelectionListener);\r
383 \r
384         fCheckBoxes.add(checkBox);\r
385         return checkBox;\r
386     }\r
387 \r
388     private Text addTextField(Composite parent, String label, String key) {\r
389         Label labelControl = new Label(parent, SWT.NONE);\r
390         labelControl.setText(label);\r
391         labelControl.setLayoutData(new GridData());\r
392 \r
393         Text textBox = new Text(parent, SWT.BORDER | SWT.SINGLE);\r
394         textBox.setData(key);\r
395         textBox.setLayoutData(new GridData());\r
396 \r
397         String currValue = (String) fWorkingValues.get(key);\r
398         textBox.setText(String.valueOf(getPositiveIntValue(currValue, 1)));\r
399         textBox.setTextLimit(3);\r
400         textBox.addModifyListener(fTextModifyListener);\r
401 \r
402         GridData gd = new GridData();\r
403         gd.widthHint = convertWidthInCharsToPixels(5);\r
404         textBox.setLayoutData(gd);\r
405 \r
406         fTextBoxes.add(textBox);\r
407         return textBox;\r
408     }\r
409 \r
410     private void controlChanged(Button button) {\r
411         ControlData data = (ControlData) button.getData();\r
412         boolean selection = button.getSelection();\r
413         String newValue = data.getValue(selection);\r
414         fWorkingValues.put(data.getKey(), newValue);\r
415         updatePreview();\r
416 \r
417         if (PREF_TAB_CHAR.equals(data.getKey())) {\r
418             updateStatus(new StatusInfo());\r
419             if (selection) {\r
420                 fTabSizeTextBox.setText((String) fWorkingValues\r
421                         .get(PREF_TAB_SIZE));\r
422             }\r
423         }\r
424     }\r
425 \r
426     private void textChanged(Text textControl) {\r
427         String key = (String) textControl.getData();\r
428         String number = textControl.getText();\r
429         IStatus status = validatePositiveNumber(number);\r
430         if (!status.matches(IStatus.ERROR)) {\r
431             fWorkingValues.put(key, number);\r
432         }\r
433         // if (PREF_TAB_SIZE.equals(key)) {\r
434         // fSourceViewer.getTextWidget().setTabs(getPositiveIntValue(number,\r
435         // 0));\r
436         // }\r
437         updateStatus(status);\r
438         updatePreview();\r
439     }\r
440 \r
441     /*\r
442      * @see IPreferencePage#performOk()\r
443      */\r
444     public boolean performOk() {\r
445         String[] allKeys = getAllKeys();\r
446         // preserve other options\r
447         // store in JCore\r
448         Hashtable actualOptions = JavaCore.getOptions();\r
449         for (int i = 0; i < allKeys.length; i++) {\r
450             String key = allKeys[i];\r
451             String val = (String) fWorkingValues.get(key);\r
452             actualOptions.put(key, val);\r
453         }\r
454         JavaCore.setOptions(actualOptions);\r
455         PHPeclipsePlugin.getDefault().savePluginPreferences();\r
456         return super.performOk();\r
457     }\r
458 \r
459     /*\r
460      * @see PreferencePage#performDefaults()\r
461      */\r
462     protected void performDefaults() {\r
463         fWorkingValues = JavaCore.getDefaultOptions();\r
464         updateControls();\r
465         super.performDefaults();\r
466     }\r
467 \r
468     private String loadPreviewFile(String filename) {\r
469         String separator = System.getProperty("line.separator"); //$NON-NLS-1$\r
470         StringBuffer btxt = new StringBuffer(512);\r
471         BufferedReader rin = null;\r
472         try {\r
473             rin = new BufferedReader(new InputStreamReader(getClass()\r
474                     .getResourceAsStream(filename)));\r
475             String line;\r
476             while ((line = rin.readLine()) != null) {\r
477                 btxt.append(line);\r
478                 btxt.append(separator);\r
479             }\r
480         } catch (IOException io) {\r
481             PHPeclipsePlugin.log(io);\r
482         } finally {\r
483             if (rin != null) {\r
484                 try {\r
485                     rin.close();\r
486                 } catch (IOException e) {\r
487                 }\r
488             }\r
489         }\r
490         return btxt.toString();\r
491     }\r
492 \r
493     private void updatePreview() {\r
494         ICodeFormatter formatter = ToolFactory\r
495                 .createDefaultCodeFormatter(fWorkingValues);\r
496         fPreviewDocument.set(formatter.format(fPreviewText, 0, null, "\n")); //$NON-NLS-1$\r
497     }\r
498 \r
499     private void updateControls() {\r
500         // update the UI\r
501         for (int i = fCheckBoxes.size() - 1; i >= 0; i--) {\r
502             Button curr = (Button) fCheckBoxes.get(i);\r
503             ControlData data = (ControlData) curr.getData();\r
504 \r
505             String currValue = (String) fWorkingValues.get(data.getKey());\r
506             curr.setSelection(data.getSelection(currValue) == 0);\r
507         }\r
508         for (int i = fTextBoxes.size() - 1; i >= 0; i--) {\r
509             Text curr = (Text) fTextBoxes.get(i);\r
510             String key = (String) curr.getData();\r
511             String currValue = (String) fWorkingValues.get(key);\r
512             curr.setText(currValue);\r
513         }\r
514     }\r
515 \r
516     private IStatus validatePositiveNumber(String number) {\r
517         StatusInfo status = new StatusInfo();\r
518         if (number.length() == 0) {\r
519             status.setError(PHPUIMessages\r
520                     .getString("CodeFormatterPreferencePage.empty_input")); //$NON-NLS-1$\r
521         } else {\r
522             try {\r
523                 int value = Integer.parseInt(number);\r
524                 if (value < 0) {\r
525                     status\r
526                             .setError(PHPUIMessages\r
527                                     .getFormattedString(\r
528                                             "CodeFormatterPreferencePage.invalid_input", number)); //$NON-NLS-1$\r
529                 }\r
530             } catch (NumberFormatException e) {\r
531                 status.setError(PHPUIMessages.getFormattedString(\r
532                         "CodeFormatterPreferencePage.invalid_input", number)); //$NON-NLS-1$\r
533             }\r
534         }\r
535         return status;\r
536     }\r
537 \r
538     private void updateStatus(IStatus status) {\r
539         if (!status.matches(IStatus.ERROR)) {\r
540             // look if there are more severe errors\r
541             for (int i = 0; i < fTextBoxes.size(); i++) {\r
542                 Text curr = (Text) fTextBoxes.get(i);\r
543                 if (!(curr == fTabSizeTextBox && usesTabs())) {\r
544                     IStatus currStatus = validatePositiveNumber(curr.getText());\r
545                     status = StatusUtil.getMoreSevere(currStatus, status);\r
546                 }\r
547             }\r
548         }\r
549         setValid(!status.matches(IStatus.ERROR));\r
550         StatusUtil.applyToStatusLine(this, status);\r
551     }\r
552 \r
553     private boolean usesTabs() {\r
554         return TAB.equals(fWorkingValues.get(PREF_TAB_CHAR));\r
555     }\r
556 \r
557 }\r