Chris@0
|
1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
|
Chris@0
|
2
|
Chris@0
|
3 /*
|
Chris@0
|
4 Sonic Visualiser
|
Chris@0
|
5 An audio file viewer and annotation editor.
|
Chris@0
|
6 Centre for Digital Music, Queen Mary, University of London.
|
Chris@77
|
7 This file copyright 2006 Chris Cannam and QMUL.
|
Chris@0
|
8
|
Chris@0
|
9 This program is free software; you can redistribute it and/or
|
Chris@0
|
10 modify it under the terms of the GNU General Public License as
|
Chris@0
|
11 published by the Free Software Foundation; either version 2 of the
|
Chris@0
|
12 License, or (at your option) any later version. See the file
|
Chris@0
|
13 COPYING included with this distribution for more information.
|
Chris@0
|
14 */
|
Chris@0
|
15
|
Chris@0
|
16 #ifndef _MAIN_WINDOW_H_
|
Chris@0
|
17 #define _MAIN_WINDOW_H_
|
Chris@0
|
18
|
Chris@0
|
19 #include <QFrame>
|
Chris@0
|
20 #include <QString>
|
Chris@85
|
21 #include <QUrl>
|
Chris@0
|
22 #include <QMainWindow>
|
Chris@0
|
23 #include <QPointer>
|
Chris@0
|
24
|
Chris@0
|
25 #include "base/Command.h"
|
Chris@1
|
26 #include "view/ViewManager.h"
|
Chris@0
|
27 #include "base/PropertyContainer.h"
|
Chris@34
|
28 #include "base/RecentFiles.h"
|
Chris@0
|
29 #include "layer/LayerFactory.h"
|
Chris@0
|
30 #include "transform/Transform.h"
|
Chris@1
|
31 #include "document/SVFileReader.h"
|
Chris@88
|
32 #include "data/fileio/FileFinder.h"
|
Chris@0
|
33 #include <map>
|
Chris@0
|
34
|
Chris@0
|
35 class Document;
|
Chris@0
|
36 class PaneStack;
|
Chris@0
|
37 class Pane;
|
Chris@0
|
38 class View;
|
Chris@0
|
39 class Fader;
|
Chris@65
|
40 class Overview;
|
Chris@0
|
41 class Layer;
|
Chris@0
|
42 class WaveformLayer;
|
Chris@0
|
43 class WaveFileModel;
|
Chris@0
|
44 class AudioCallbackPlaySource;
|
Chris@0
|
45 class AudioCallbackPlayTarget;
|
Chris@0
|
46 class CommandHistory;
|
Chris@0
|
47 class QMenu;
|
Chris@0
|
48 class AudioDial;
|
Chris@0
|
49 class QLabel;
|
Chris@16
|
50 class QCheckBox;
|
Chris@0
|
51 class PreferencesDialog;
|
Chris@177
|
52 class QTreeView;
|
Chris@26
|
53 class QPushButton;
|
Chris@69
|
54 class OSCQueue;
|
Chris@69
|
55 class OSCMessage;
|
Chris@162
|
56 class KeyReference;
|
Chris@189
|
57 class Labeller;
|
Chris@0
|
58
|
Chris@0
|
59
|
Chris@0
|
60 class MainWindow : public QMainWindow
|
Chris@0
|
61 {
|
Chris@0
|
62 Q_OBJECT
|
Chris@0
|
63
|
Chris@0
|
64 public:
|
Chris@70
|
65 MainWindow(bool withAudioOutput = true,
|
Chris@70
|
66 bool withOSCSupport = true);
|
Chris@0
|
67 virtual ~MainWindow();
|
Chris@0
|
68
|
Chris@0
|
69 enum AudioFileOpenMode {
|
Chris@0
|
70 ReplaceMainModel,
|
Chris@0
|
71 CreateAdditionalModel,
|
Chris@0
|
72 AskUser
|
Chris@0
|
73 };
|
Chris@0
|
74
|
Chris@82
|
75 enum FileOpenStatus {
|
Chris@82
|
76 FileOpenSucceeded,
|
Chris@82
|
77 FileOpenFailed,
|
Chris@82
|
78 FileOpenCancelled
|
Chris@82
|
79 };
|
Chris@82
|
80
|
Chris@82
|
81 FileOpenStatus openSomeFile(QString path, AudioFileOpenMode = AskUser);
|
Chris@82
|
82 FileOpenStatus openAudioFile(QString path, AudioFileOpenMode = AskUser);
|
Chris@180
|
83 FileOpenStatus openPlaylistFile(QString path, AudioFileOpenMode = AskUser);
|
Chris@82
|
84 FileOpenStatus openLayerFile(QString path);
|
Chris@82
|
85 FileOpenStatus openSessionFile(QString path);
|
Chris@180
|
86 FileOpenStatus openURL(QUrl url, AudioFileOpenMode = AskUser);
|
Chris@180
|
87 FileOpenStatus openURL(QString url, AudioFileOpenMode = AskUser);
|
Chris@82
|
88
|
Chris@0
|
89 bool saveSessionFile(QString path);
|
Chris@11
|
90 bool commitData(bool mayAskUser); // on session shutdown
|
Chris@0
|
91
|
Chris@0
|
92 signals:
|
Chris@0
|
93 // Used to toggle the availability of menu actions
|
Chris@0
|
94 void canAddPane(bool);
|
Chris@0
|
95 void canDeleteCurrentPane(bool);
|
Chris@0
|
96 void canAddLayer(bool);
|
Chris@0
|
97 void canImportMoreAudio(bool);
|
Chris@0
|
98 void canImportLayer(bool);
|
Chris@0
|
99 void canExportAudio(bool);
|
Chris@0
|
100 void canExportLayer(bool);
|
Chris@121
|
101 void canExportImage(bool);
|
Chris@0
|
102 void canRenameLayer(bool);
|
Chris@0
|
103 void canEditLayer(bool);
|
Chris@169
|
104 void canMeasureLayer(bool);
|
Chris@0
|
105 void canSelect(bool);
|
Chris@0
|
106 void canClearSelection(bool);
|
Chris@0
|
107 void canEditSelection(bool);
|
Chris@164
|
108 void canDeleteSelection(bool);
|
Chris@0
|
109 void canPaste(bool);
|
Chris@0
|
110 void canInsertInstant(bool);
|
Chris@81
|
111 void canInsertInstantsAtBoundaries(bool);
|
Chris@189
|
112 void canRenumberInstants(bool);
|
Chris@0
|
113 void canDeleteCurrentLayer(bool);
|
Chris@0
|
114 void canZoom(bool);
|
Chris@0
|
115 void canScroll(bool);
|
Chris@0
|
116 void canPlay(bool);
|
Chris@0
|
117 void canFfwd(bool);
|
Chris@0
|
118 void canRewind(bool);
|
Chris@0
|
119 void canPlaySelection(bool);
|
Chris@155
|
120 void canSpeedUpPlayback(bool);
|
Chris@155
|
121 void canSlowDownPlayback(bool);
|
Chris@155
|
122 void canChangePlaybackSpeed(bool);
|
Chris@0
|
123 void canSave(bool);
|
Chris@0
|
124
|
Chris@118
|
125 public slots:
|
Chris@118
|
126 void preferenceChanged(PropertyContainer::PropertyName);
|
Chris@118
|
127
|
Chris@0
|
128 protected slots:
|
Chris@0
|
129 void openSession();
|
Chris@0
|
130 void importAudio();
|
Chris@0
|
131 void importMoreAudio();
|
Chris@0
|
132 void openSomething();
|
Chris@86
|
133 void openLocation();
|
Chris@0
|
134 void openRecentFile();
|
Chris@0
|
135 void exportAudio();
|
Chris@0
|
136 void importLayer();
|
Chris@0
|
137 void exportLayer();
|
Chris@121
|
138 void exportImage();
|
Chris@0
|
139 void saveSession();
|
Chris@0
|
140 void saveSessionAs();
|
Chris@0
|
141 void newSession();
|
Chris@0
|
142 void closeSession();
|
Chris@0
|
143 void preferences();
|
Chris@0
|
144
|
Chris@0
|
145 void zoomIn();
|
Chris@0
|
146 void zoomOut();
|
Chris@0
|
147 void zoomToFit();
|
Chris@0
|
148 void zoomDefault();
|
Chris@0
|
149 void scrollLeft();
|
Chris@0
|
150 void scrollRight();
|
Chris@0
|
151 void jumpLeft();
|
Chris@0
|
152 void jumpRight();
|
Chris@0
|
153
|
Chris@0
|
154 void showNoOverlays();
|
Chris@90
|
155 void showMinimalOverlays();
|
Chris@90
|
156 void showStandardOverlays();
|
Chris@90
|
157 void showAllOverlays();
|
Chris@7
|
158
|
Chris@7
|
159 void toggleZoomWheels();
|
Chris@72
|
160 void togglePropertyBoxes();
|
Chris@90
|
161 void toggleStatusBar();
|
Chris@0
|
162
|
Chris@0
|
163 void play();
|
Chris@0
|
164 void ffwd();
|
Chris@0
|
165 void ffwdEnd();
|
Chris@0
|
166 void rewind();
|
Chris@0
|
167 void rewindStart();
|
Chris@0
|
168 void stop();
|
Chris@0
|
169
|
Chris@0
|
170 void addPane();
|
Chris@0
|
171 void addLayer();
|
Chris@0
|
172 void deleteCurrentPane();
|
Chris@0
|
173 void renameCurrentLayer();
|
Chris@0
|
174 void deleteCurrentLayer();
|
Chris@0
|
175
|
Chris@0
|
176 void playLoopToggled();
|
Chris@0
|
177 void playSelectionToggled();
|
Chris@180
|
178 void playSoloToggled();
|
Chris@0
|
179 void playSpeedChanged(int);
|
Chris@16
|
180 void playSharpenToggled();
|
Chris@26
|
181 void playMonoToggled();
|
Chris@155
|
182 void speedUpPlayback();
|
Chris@155
|
183 void slowDownPlayback();
|
Chris@155
|
184 void restoreNormalPlayback();
|
Chris@0
|
185 void sampleRateMismatch(size_t, size_t, bool);
|
Chris@42
|
186 void audioOverloadPluginDisabled();
|
Chris@0
|
187
|
Chris@116
|
188 void playbackFrameChanged(unsigned long);
|
Chris@116
|
189 void globalCentreFrameChanged(unsigned long);
|
Chris@116
|
190 void viewCentreFrameChanged(View *, unsigned long);
|
Chris@116
|
191 void viewZoomLevelChanged(View *, unsigned long, bool);
|
Chris@0
|
192 void outputLevelsChanged(float, float);
|
Chris@0
|
193
|
Chris@0
|
194 void currentPaneChanged(Pane *);
|
Chris@0
|
195 void currentLayerChanged(Pane *, Layer *);
|
Chris@0
|
196
|
Chris@0
|
197 void toolNavigateSelected();
|
Chris@0
|
198 void toolSelectSelected();
|
Chris@0
|
199 void toolEditSelected();
|
Chris@0
|
200 void toolDrawSelected();
|
Chris@151
|
201 void toolMeasureSelected();
|
Chris@0
|
202
|
Chris@0
|
203 void selectAll();
|
Chris@0
|
204 void selectToStart();
|
Chris@0
|
205 void selectToEnd();
|
Chris@0
|
206 void selectVisible();
|
Chris@0
|
207 void clearSelection();
|
Chris@0
|
208 void cut();
|
Chris@0
|
209 void copy();
|
Chris@0
|
210 void paste();
|
Chris@0
|
211 void deleteSelected();
|
Chris@0
|
212 void insertInstant();
|
Chris@81
|
213 void insertInstantAt(size_t);
|
Chris@81
|
214 void insertInstantsAtBoundaries();
|
Chris@189
|
215 void setInstantsNumbering();
|
Chris@189
|
216 void setInstantsCounterCycle();
|
Chris@192
|
217 void resetInstantsCounters();
|
Chris@189
|
218 void renumberInstants();
|
Chris@0
|
219
|
Chris@0
|
220 void documentModified();
|
Chris@0
|
221 void documentRestored();
|
Chris@0
|
222
|
Chris@0
|
223 void updateMenuStates();
|
Chris@0
|
224 void updateDescriptionLabel();
|
Chris@0
|
225
|
Chris@0
|
226 void layerAdded(Layer *);
|
Chris@0
|
227 void layerRemoved(Layer *);
|
Chris@0
|
228 void layerAboutToBeDeleted(Layer *);
|
Chris@0
|
229 void layerInAView(Layer *, bool);
|
Chris@0
|
230
|
Chris@0
|
231 void mainModelChanged(WaveFileModel *);
|
Chris@0
|
232 void modelAdded(Model *);
|
Chris@0
|
233 void modelAboutToBeDeleted(Model *);
|
Chris@0
|
234
|
Chris@0
|
235 void modelGenerationFailed(QString);
|
Chris@0
|
236 void modelRegenerationFailed(QString, QString);
|
Chris@0
|
237
|
Chris@0
|
238 void rightButtonMenuRequested(Pane *, QPoint point);
|
Chris@0
|
239
|
Chris@73
|
240 void propertyStacksResized();
|
Chris@73
|
241
|
Chris@0
|
242 void setupRecentFilesMenu();
|
Chris@34
|
243 void setupRecentTransformsMenu();
|
Chris@0
|
244
|
Chris@0
|
245 void showLayerTree();
|
Chris@0
|
246
|
Chris@69
|
247 void pollOSC();
|
Chris@69
|
248 void handleOSCMessage(const OSCMessage &);
|
Chris@69
|
249
|
Chris@90
|
250 void mouseEnteredWidget();
|
Chris@90
|
251 void mouseLeftWidget();
|
Chris@116
|
252 void contextHelpChanged(const QString &);
|
Chris@117
|
253 void inProgressSelectionChanged();
|
Chris@90
|
254
|
Chris@0
|
255 void website();
|
Chris@0
|
256 void help();
|
Chris@0
|
257 void about();
|
Chris@162
|
258 void keyReference();
|
Chris@0
|
259
|
Chris@0
|
260 protected:
|
Chris@0
|
261 QString m_sessionFile;
|
Chris@0
|
262 QString m_audioFile;
|
Chris@0
|
263 Document *m_document;
|
Chris@0
|
264
|
Chris@0
|
265 QLabel *m_descriptionLabel;
|
Chris@0
|
266 PaneStack *m_paneStack;
|
Chris@0
|
267 ViewManager *m_viewManager;
|
Chris@65
|
268 Overview *m_overview;
|
Chris@0
|
269 Fader *m_fader;
|
Chris@0
|
270 AudioDial *m_playSpeed;
|
Chris@26
|
271 QPushButton *m_playSharpen;
|
Chris@26
|
272 QPushButton *m_playMono;
|
Chris@0
|
273 WaveformLayer *m_panLayer;
|
Chris@0
|
274 Layer *m_timeRulerLayer;
|
Chris@0
|
275
|
Chris@46
|
276 bool m_audioOutput;
|
Chris@0
|
277 AudioCallbackPlaySource *m_playSource;
|
Chris@0
|
278 AudioCallbackPlayTarget *m_playTarget;
|
Chris@0
|
279
|
Chris@69
|
280 OSCQueue *m_oscQueue;
|
Chris@69
|
281
|
Chris@34
|
282 RecentFiles m_recentFiles;
|
Chris@34
|
283 RecentFiles m_recentTransforms;
|
Chris@34
|
284
|
Chris@0
|
285 bool m_mainMenusCreated;
|
Chris@0
|
286 QMenu *m_paneMenu;
|
Chris@0
|
287 QMenu *m_layerMenu;
|
Chris@34
|
288 QMenu *m_transformsMenu;
|
Chris@155
|
289 QMenu *m_playbackMenu;
|
Chris@0
|
290 QMenu *m_existingLayersMenu;
|
Chris@95
|
291 QMenu *m_sliceMenu;
|
Chris@0
|
292 QMenu *m_recentFilesMenu;
|
Chris@34
|
293 QMenu *m_recentTransformsMenu;
|
Chris@0
|
294 QMenu *m_rightButtonMenu;
|
Chris@0
|
295 QMenu *m_rightButtonLayerMenu;
|
Chris@34
|
296 QMenu *m_rightButtonTransformsMenu;
|
Chris@155
|
297 QMenu *m_rightButtonPlaybackMenu;
|
Chris@155
|
298
|
Chris@164
|
299 QAction *m_deleteSelectedAction;
|
Chris@155
|
300 QAction *m_ffwdAction;
|
Chris@155
|
301 QAction *m_rwdAction;
|
Chris@0
|
302
|
Chris@0
|
303 bool m_documentModified;
|
Chris@70
|
304 bool m_openingAudioFile;
|
Chris@70
|
305 bool m_abandoning;
|
Chris@0
|
306
|
Chris@189
|
307 Labeller *m_labeller;
|
Chris@189
|
308
|
Chris@116
|
309 int m_lastPlayStatusSec;
|
Chris@116
|
310 mutable QString m_myStatusMessage;
|
Chris@116
|
311
|
Chris@0
|
312 QPointer<PreferencesDialog> m_preferencesDialog;
|
Chris@177
|
313 QPointer<QTreeView> m_layerTreeView;
|
Chris@0
|
314
|
Chris@180
|
315 bool m_initialDarkBackground;
|
Chris@180
|
316
|
Chris@162
|
317 KeyReference *m_keyReference;
|
Chris@162
|
318
|
Chris@0
|
319 WaveFileModel *getMainModel();
|
Chris@116
|
320 const WaveFileModel *getMainModel() const;
|
Chris@0
|
321 void createDocument();
|
Chris@0
|
322
|
Chris@0
|
323 struct PaneConfiguration {
|
Chris@0
|
324 PaneConfiguration(LayerFactory::LayerType _layer
|
Chris@0
|
325 = LayerFactory::TimeRuler,
|
Chris@66
|
326 Model *_source = 0,
|
Chris@0
|
327 int _channel = -1) :
|
Chris@66
|
328 layer(_layer), sourceModel(_source), channel(_channel) { }
|
Chris@0
|
329 LayerFactory::LayerType layer;
|
Chris@66
|
330 Model *sourceModel;
|
Chris@0
|
331 int channel;
|
Chris@0
|
332 };
|
Chris@0
|
333
|
Chris@0
|
334 typedef std::map<QAction *, PaneConfiguration> PaneActionMap;
|
Chris@0
|
335 PaneActionMap m_paneActions;
|
Chris@0
|
336
|
Chris@107
|
337 typedef std::map<QAction *, TransformId> TransformActionMap;
|
Chris@34
|
338 TransformActionMap m_transformActions;
|
Chris@34
|
339
|
Chris@107
|
340 typedef std::map<TransformId, QAction *> TransformActionReverseMap;
|
Chris@34
|
341 TransformActionReverseMap m_transformActionsReverse;
|
Chris@0
|
342
|
Chris@0
|
343 typedef std::map<QAction *, LayerFactory::LayerType> LayerActionMap;
|
Chris@0
|
344 LayerActionMap m_layerActions;
|
Chris@0
|
345
|
Chris@0
|
346 typedef std::map<QAction *, Layer *> ExistingLayerActionMap;
|
Chris@0
|
347 ExistingLayerActionMap m_existingLayerActions;
|
Chris@95
|
348 ExistingLayerActionMap m_sliceActions;
|
Chris@0
|
349
|
Chris@0
|
350 typedef std::map<ViewManager::ToolMode, QAction *> ToolActionMap;
|
Chris@0
|
351 ToolActionMap m_toolActions;
|
Chris@0
|
352
|
Chris@189
|
353 typedef std::map<QAction *, int> NumberingActionMap;
|
Chris@189
|
354 NumberingActionMap m_numberingActions;
|
Chris@189
|
355
|
Chris@0
|
356 void setupMenus();
|
Chris@66
|
357 void setupFileMenu();
|
Chris@66
|
358 void setupEditMenu();
|
Chris@66
|
359 void setupViewMenu();
|
Chris@66
|
360 void setupPaneAndLayerMenus();
|
Chris@66
|
361 void setupTransformsMenu();
|
Chris@66
|
362 void setupHelpMenu();
|
Chris@95
|
363 void setupExistingLayersMenus();
|
Chris@0
|
364 void setupToolbars();
|
Chris@66
|
365
|
Chris@0
|
366 Pane *addPaneToStack();
|
Chris@0
|
367
|
Chris@69
|
368 void addPane(const PaneConfiguration &configuration, QString text);
|
Chris@69
|
369
|
Chris@155
|
370 Layer *getSnapLayer() const;
|
Chris@155
|
371
|
Chris@0
|
372 class PaneCallback : public SVFileReaderPaneCallback
|
Chris@0
|
373 {
|
Chris@0
|
374 public:
|
Chris@0
|
375 PaneCallback(MainWindow *mw) : m_mw(mw) { }
|
Chris@0
|
376 virtual Pane *addPane() { return m_mw->addPaneToStack(); }
|
Chris@0
|
377 virtual void setWindowSize(int width, int height) {
|
Chris@0
|
378 m_mw->resize(width, height);
|
Chris@0
|
379 }
|
Chris@0
|
380 virtual void addSelection(int start, int end) {
|
Chris@0
|
381 m_mw->m_viewManager->addSelection(Selection(start, end));
|
Chris@0
|
382 }
|
Chris@0
|
383 protected:
|
Chris@0
|
384 MainWindow *m_mw;
|
Chris@0
|
385 };
|
Chris@0
|
386
|
Chris@0
|
387 class AddPaneCommand : public Command
|
Chris@0
|
388 {
|
Chris@0
|
389 public:
|
Chris@0
|
390 AddPaneCommand(MainWindow *mw);
|
Chris@0
|
391 virtual ~AddPaneCommand();
|
Chris@0
|
392
|
Chris@0
|
393 virtual void execute();
|
Chris@0
|
394 virtual void unexecute();
|
Chris@0
|
395 virtual QString getName() const;
|
Chris@0
|
396
|
Chris@0
|
397 Pane *getPane() { return m_pane; }
|
Chris@0
|
398
|
Chris@0
|
399 protected:
|
Chris@0
|
400 MainWindow *m_mw;
|
Chris@0
|
401 Pane *m_pane; // Main window owns this, but I determine its lifespan
|
Chris@0
|
402 Pane *m_prevCurrentPane; // I don't own this
|
Chris@0
|
403 bool m_added;
|
Chris@0
|
404 };
|
Chris@0
|
405
|
Chris@0
|
406 class RemovePaneCommand : public Command
|
Chris@0
|
407 {
|
Chris@0
|
408 public:
|
Chris@0
|
409 RemovePaneCommand(MainWindow *mw, Pane *pane);
|
Chris@0
|
410 virtual ~RemovePaneCommand();
|
Chris@0
|
411
|
Chris@0
|
412 virtual void execute();
|
Chris@0
|
413 virtual void unexecute();
|
Chris@0
|
414 virtual QString getName() const;
|
Chris@0
|
415
|
Chris@0
|
416 protected:
|
Chris@0
|
417 MainWindow *m_mw;
|
Chris@0
|
418 Pane *m_pane; // Main window owns this, but I determine its lifespan
|
Chris@0
|
419 Pane *m_prevCurrentPane; // I don't own this
|
Chris@0
|
420 bool m_added;
|
Chris@0
|
421 };
|
Chris@0
|
422
|
Chris@0
|
423 virtual void closeEvent(QCloseEvent *e);
|
Chris@0
|
424 bool checkSaveModified();
|
Chris@0
|
425
|
Chris@86
|
426 FileOpenStatus openSomeFile(QString path, QString location,
|
Chris@86
|
427 AudioFileOpenMode = AskUser);
|
Chris@86
|
428 FileOpenStatus openAudioFile(QString path, QString location,
|
Chris@86
|
429 AudioFileOpenMode = AskUser);
|
Chris@180
|
430 FileOpenStatus openPlaylistFile(QString path, QString location,
|
Chris@180
|
431 AudioFileOpenMode = AskUser);
|
Chris@86
|
432 FileOpenStatus openLayerFile(QString path, QString location);
|
Chris@86
|
433 FileOpenStatus openSessionFile(QString path, QString location);
|
Chris@86
|
434
|
Chris@88
|
435 QString getOpenFileName(FileFinder::FileType type);
|
Chris@88
|
436 QString getSaveFileName(FileFinder::FileType type);
|
Chris@88
|
437 void registerLastOpenedFilePath(FileFinder::FileType type, QString path);
|
Chris@81
|
438
|
Chris@0
|
439 void createPlayTarget();
|
Chris@0
|
440
|
Chris@0
|
441 void openHelpUrl(QString url);
|
Chris@0
|
442
|
Chris@116
|
443 void updateVisibleRangeDisplay(Pane *p) const;
|
Chris@116
|
444
|
Chris@0
|
445 void toXml(QTextStream &stream);
|
Chris@0
|
446 };
|
Chris@0
|
447
|
Chris@0
|
448
|
Chris@0
|
449 #endif
|