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@189
|
217 void renumberInstants();
|
Chris@0
|
218
|
Chris@0
|
219 void documentModified();
|
Chris@0
|
220 void documentRestored();
|
Chris@0
|
221
|
Chris@0
|
222 void updateMenuStates();
|
Chris@0
|
223 void updateDescriptionLabel();
|
Chris@0
|
224
|
Chris@0
|
225 void layerAdded(Layer *);
|
Chris@0
|
226 void layerRemoved(Layer *);
|
Chris@0
|
227 void layerAboutToBeDeleted(Layer *);
|
Chris@0
|
228 void layerInAView(Layer *, bool);
|
Chris@0
|
229
|
Chris@0
|
230 void mainModelChanged(WaveFileModel *);
|
Chris@0
|
231 void modelAdded(Model *);
|
Chris@0
|
232 void modelAboutToBeDeleted(Model *);
|
Chris@0
|
233
|
Chris@0
|
234 void modelGenerationFailed(QString);
|
Chris@0
|
235 void modelRegenerationFailed(QString, QString);
|
Chris@0
|
236
|
Chris@0
|
237 void rightButtonMenuRequested(Pane *, QPoint point);
|
Chris@0
|
238
|
Chris@73
|
239 void propertyStacksResized();
|
Chris@73
|
240
|
Chris@0
|
241 void setupRecentFilesMenu();
|
Chris@34
|
242 void setupRecentTransformsMenu();
|
Chris@0
|
243
|
Chris@0
|
244 void showLayerTree();
|
Chris@0
|
245
|
Chris@69
|
246 void pollOSC();
|
Chris@69
|
247 void handleOSCMessage(const OSCMessage &);
|
Chris@69
|
248
|
Chris@90
|
249 void mouseEnteredWidget();
|
Chris@90
|
250 void mouseLeftWidget();
|
Chris@116
|
251 void contextHelpChanged(const QString &);
|
Chris@117
|
252 void inProgressSelectionChanged();
|
Chris@90
|
253
|
Chris@0
|
254 void website();
|
Chris@0
|
255 void help();
|
Chris@0
|
256 void about();
|
Chris@162
|
257 void keyReference();
|
Chris@0
|
258
|
Chris@0
|
259 protected:
|
Chris@0
|
260 QString m_sessionFile;
|
Chris@0
|
261 QString m_audioFile;
|
Chris@0
|
262 Document *m_document;
|
Chris@0
|
263
|
Chris@0
|
264 QLabel *m_descriptionLabel;
|
Chris@0
|
265 PaneStack *m_paneStack;
|
Chris@0
|
266 ViewManager *m_viewManager;
|
Chris@65
|
267 Overview *m_overview;
|
Chris@0
|
268 Fader *m_fader;
|
Chris@0
|
269 AudioDial *m_playSpeed;
|
Chris@26
|
270 QPushButton *m_playSharpen;
|
Chris@26
|
271 QPushButton *m_playMono;
|
Chris@0
|
272 WaveformLayer *m_panLayer;
|
Chris@0
|
273 Layer *m_timeRulerLayer;
|
Chris@0
|
274
|
Chris@46
|
275 bool m_audioOutput;
|
Chris@0
|
276 AudioCallbackPlaySource *m_playSource;
|
Chris@0
|
277 AudioCallbackPlayTarget *m_playTarget;
|
Chris@0
|
278
|
Chris@69
|
279 OSCQueue *m_oscQueue;
|
Chris@69
|
280
|
Chris@34
|
281 RecentFiles m_recentFiles;
|
Chris@34
|
282 RecentFiles m_recentTransforms;
|
Chris@34
|
283
|
Chris@0
|
284 bool m_mainMenusCreated;
|
Chris@0
|
285 QMenu *m_paneMenu;
|
Chris@0
|
286 QMenu *m_layerMenu;
|
Chris@34
|
287 QMenu *m_transformsMenu;
|
Chris@155
|
288 QMenu *m_playbackMenu;
|
Chris@0
|
289 QMenu *m_existingLayersMenu;
|
Chris@95
|
290 QMenu *m_sliceMenu;
|
Chris@0
|
291 QMenu *m_recentFilesMenu;
|
Chris@34
|
292 QMenu *m_recentTransformsMenu;
|
Chris@0
|
293 QMenu *m_rightButtonMenu;
|
Chris@0
|
294 QMenu *m_rightButtonLayerMenu;
|
Chris@34
|
295 QMenu *m_rightButtonTransformsMenu;
|
Chris@155
|
296 QMenu *m_rightButtonPlaybackMenu;
|
Chris@155
|
297
|
Chris@164
|
298 QAction *m_deleteSelectedAction;
|
Chris@155
|
299 QAction *m_ffwdAction;
|
Chris@155
|
300 QAction *m_rwdAction;
|
Chris@0
|
301
|
Chris@0
|
302 bool m_documentModified;
|
Chris@70
|
303 bool m_openingAudioFile;
|
Chris@70
|
304 bool m_abandoning;
|
Chris@0
|
305
|
Chris@189
|
306 Labeller *m_labeller;
|
Chris@189
|
307
|
Chris@116
|
308 int m_lastPlayStatusSec;
|
Chris@116
|
309 mutable QString m_myStatusMessage;
|
Chris@116
|
310
|
Chris@0
|
311 QPointer<PreferencesDialog> m_preferencesDialog;
|
Chris@177
|
312 QPointer<QTreeView> m_layerTreeView;
|
Chris@0
|
313
|
Chris@180
|
314 bool m_initialDarkBackground;
|
Chris@180
|
315
|
Chris@162
|
316 KeyReference *m_keyReference;
|
Chris@162
|
317
|
Chris@0
|
318 WaveFileModel *getMainModel();
|
Chris@116
|
319 const WaveFileModel *getMainModel() const;
|
Chris@0
|
320 void createDocument();
|
Chris@0
|
321
|
Chris@0
|
322 struct PaneConfiguration {
|
Chris@0
|
323 PaneConfiguration(LayerFactory::LayerType _layer
|
Chris@0
|
324 = LayerFactory::TimeRuler,
|
Chris@66
|
325 Model *_source = 0,
|
Chris@0
|
326 int _channel = -1) :
|
Chris@66
|
327 layer(_layer), sourceModel(_source), channel(_channel) { }
|
Chris@0
|
328 LayerFactory::LayerType layer;
|
Chris@66
|
329 Model *sourceModel;
|
Chris@0
|
330 int channel;
|
Chris@0
|
331 };
|
Chris@0
|
332
|
Chris@0
|
333 typedef std::map<QAction *, PaneConfiguration> PaneActionMap;
|
Chris@0
|
334 PaneActionMap m_paneActions;
|
Chris@0
|
335
|
Chris@107
|
336 typedef std::map<QAction *, TransformId> TransformActionMap;
|
Chris@34
|
337 TransformActionMap m_transformActions;
|
Chris@34
|
338
|
Chris@107
|
339 typedef std::map<TransformId, QAction *> TransformActionReverseMap;
|
Chris@34
|
340 TransformActionReverseMap m_transformActionsReverse;
|
Chris@0
|
341
|
Chris@0
|
342 typedef std::map<QAction *, LayerFactory::LayerType> LayerActionMap;
|
Chris@0
|
343 LayerActionMap m_layerActions;
|
Chris@0
|
344
|
Chris@0
|
345 typedef std::map<QAction *, Layer *> ExistingLayerActionMap;
|
Chris@0
|
346 ExistingLayerActionMap m_existingLayerActions;
|
Chris@95
|
347 ExistingLayerActionMap m_sliceActions;
|
Chris@0
|
348
|
Chris@0
|
349 typedef std::map<ViewManager::ToolMode, QAction *> ToolActionMap;
|
Chris@0
|
350 ToolActionMap m_toolActions;
|
Chris@0
|
351
|
Chris@189
|
352 typedef std::map<QAction *, int> NumberingActionMap;
|
Chris@189
|
353 NumberingActionMap m_numberingActions;
|
Chris@189
|
354
|
Chris@0
|
355 void setupMenus();
|
Chris@66
|
356 void setupFileMenu();
|
Chris@66
|
357 void setupEditMenu();
|
Chris@66
|
358 void setupViewMenu();
|
Chris@66
|
359 void setupPaneAndLayerMenus();
|
Chris@66
|
360 void setupTransformsMenu();
|
Chris@66
|
361 void setupHelpMenu();
|
Chris@95
|
362 void setupExistingLayersMenus();
|
Chris@0
|
363 void setupToolbars();
|
Chris@66
|
364
|
Chris@0
|
365 Pane *addPaneToStack();
|
Chris@0
|
366
|
Chris@69
|
367 void addPane(const PaneConfiguration &configuration, QString text);
|
Chris@69
|
368
|
Chris@155
|
369 Layer *getSnapLayer() const;
|
Chris@155
|
370
|
Chris@0
|
371 class PaneCallback : public SVFileReaderPaneCallback
|
Chris@0
|
372 {
|
Chris@0
|
373 public:
|
Chris@0
|
374 PaneCallback(MainWindow *mw) : m_mw(mw) { }
|
Chris@0
|
375 virtual Pane *addPane() { return m_mw->addPaneToStack(); }
|
Chris@0
|
376 virtual void setWindowSize(int width, int height) {
|
Chris@0
|
377 m_mw->resize(width, height);
|
Chris@0
|
378 }
|
Chris@0
|
379 virtual void addSelection(int start, int end) {
|
Chris@0
|
380 m_mw->m_viewManager->addSelection(Selection(start, end));
|
Chris@0
|
381 }
|
Chris@0
|
382 protected:
|
Chris@0
|
383 MainWindow *m_mw;
|
Chris@0
|
384 };
|
Chris@0
|
385
|
Chris@0
|
386 class AddPaneCommand : public Command
|
Chris@0
|
387 {
|
Chris@0
|
388 public:
|
Chris@0
|
389 AddPaneCommand(MainWindow *mw);
|
Chris@0
|
390 virtual ~AddPaneCommand();
|
Chris@0
|
391
|
Chris@0
|
392 virtual void execute();
|
Chris@0
|
393 virtual void unexecute();
|
Chris@0
|
394 virtual QString getName() const;
|
Chris@0
|
395
|
Chris@0
|
396 Pane *getPane() { return m_pane; }
|
Chris@0
|
397
|
Chris@0
|
398 protected:
|
Chris@0
|
399 MainWindow *m_mw;
|
Chris@0
|
400 Pane *m_pane; // Main window owns this, but I determine its lifespan
|
Chris@0
|
401 Pane *m_prevCurrentPane; // I don't own this
|
Chris@0
|
402 bool m_added;
|
Chris@0
|
403 };
|
Chris@0
|
404
|
Chris@0
|
405 class RemovePaneCommand : public Command
|
Chris@0
|
406 {
|
Chris@0
|
407 public:
|
Chris@0
|
408 RemovePaneCommand(MainWindow *mw, Pane *pane);
|
Chris@0
|
409 virtual ~RemovePaneCommand();
|
Chris@0
|
410
|
Chris@0
|
411 virtual void execute();
|
Chris@0
|
412 virtual void unexecute();
|
Chris@0
|
413 virtual QString getName() const;
|
Chris@0
|
414
|
Chris@0
|
415 protected:
|
Chris@0
|
416 MainWindow *m_mw;
|
Chris@0
|
417 Pane *m_pane; // Main window owns this, but I determine its lifespan
|
Chris@0
|
418 Pane *m_prevCurrentPane; // I don't own this
|
Chris@0
|
419 bool m_added;
|
Chris@0
|
420 };
|
Chris@0
|
421
|
Chris@0
|
422 virtual void closeEvent(QCloseEvent *e);
|
Chris@0
|
423 bool checkSaveModified();
|
Chris@0
|
424
|
Chris@86
|
425 FileOpenStatus openSomeFile(QString path, QString location,
|
Chris@86
|
426 AudioFileOpenMode = AskUser);
|
Chris@86
|
427 FileOpenStatus openAudioFile(QString path, QString location,
|
Chris@86
|
428 AudioFileOpenMode = AskUser);
|
Chris@180
|
429 FileOpenStatus openPlaylistFile(QString path, QString location,
|
Chris@180
|
430 AudioFileOpenMode = AskUser);
|
Chris@86
|
431 FileOpenStatus openLayerFile(QString path, QString location);
|
Chris@86
|
432 FileOpenStatus openSessionFile(QString path, QString location);
|
Chris@86
|
433
|
Chris@88
|
434 QString getOpenFileName(FileFinder::FileType type);
|
Chris@88
|
435 QString getSaveFileName(FileFinder::FileType type);
|
Chris@88
|
436 void registerLastOpenedFilePath(FileFinder::FileType type, QString path);
|
Chris@81
|
437
|
Chris@0
|
438 void createPlayTarget();
|
Chris@0
|
439
|
Chris@0
|
440 void openHelpUrl(QString url);
|
Chris@0
|
441
|
Chris@116
|
442 void updateVisibleRangeDisplay(Pane *p) const;
|
Chris@116
|
443
|
Chris@0
|
444 void toXml(QTextStream &stream);
|
Chris@0
|
445 };
|
Chris@0
|
446
|
Chris@0
|
447
|
Chris@0
|
448 #endif
|