openFunctions.cc 29.6 KB
Newer Older
1
/*===========================================================================*\
Jan Möbius's avatar
Jan Möbius committed
2
3
*                                                                            *
*                              OpenFlipper                                   *
Jan Möbius's avatar
Jan Möbius committed
4
*      Copyright (C) 2001-2011 by Computer Graphics Group, RWTH Aachen       *
Jan Möbius's avatar
Jan Möbius committed
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
*                           www.openflipper.org                              *
*                                                                            *
*--------------------------------------------------------------------------- *
*  This file is part of OpenFlipper.                                         *
*                                                                            *
*  OpenFlipper is free software: you can redistribute it and/or modify       *
*  it under the terms of the GNU Lesser General Public License as            *
*  published by the Free Software Foundation, either version 3 of            *
*  the License, or (at your option) any later version with the               *
*  following exceptions:                                                     *
*                                                                            *
*  If other files instantiate templates or use macros                        *
*  or inline functions from this file, or you compile this file and          *
*  link it with other files to produce an executable, this file does         *
*  not by itself cause the resulting executable to be covered by the         *
*  GNU Lesser General Public License. This exception does not however        *
*  invalidate any other reasons why the executable file might be             *
*  covered by the GNU Lesser General Public License.                         *
*                                                                            *
*  OpenFlipper is distributed in the hope that it will be useful,            *
*  but WITHOUT ANY WARRANTY; without even the implied warranty of            *
*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the             *
*  GNU Lesser General Public License for more details.                       *
*                                                                            *
*  You should have received a copy of the GNU LesserGeneral Public           *
*  License along with OpenFlipper. If not,                                   *
*  see <http://www.gnu.org/licenses/>.                                       *
*                                                                            *
33
34
35
\*===========================================================================*/

/*===========================================================================*\
Jan Möbius's avatar
Jan Möbius committed
36
37
38
39
40
*                                                                            *
*   $Revision$                                                       *
*   $LastChangedBy$                                                *
*   $Date$                     *
*                                                                            *
41
\*===========================================================================*/
Jan Möbius's avatar
 
Jan Möbius committed
42
43
44
45
46
47
48
49





#include "Core.hh"

#include <ACG/QtWidgets/QtFileDialog.hh>
Jan Möbius's avatar
Jan Möbius committed
50
#include <ACG/Scenegraph/SceneGraphAnalysis.hh>
Jan Möbius's avatar
 
Jan Möbius committed
51
52

#include "OpenFlipper/common/GlobalOptions.hh"
53
#include <OpenFlipper/common/RecentFiles.hh>
Jan Möbius's avatar
 
Jan Möbius committed
54
55
56
57
58
#include "OpenFlipper/BasePlugin/PluginFunctions.hh"

#include "OpenFlipper/widgets/loadWidget/loadWidget.hh"
#include "OpenFlipper/widgets/addEmptyWidget/addEmptyWidget.hh"

59
60
61
#include <OpenFlipper/common/Types.hh>
#include <ObjectTypes/PolyMesh/PolyMesh.hh>

62
#include <OpenFlipper/common/DataTypes.hh>
63

64
void Core::resetScenegraph( bool _resetTrackBall  ) {
Jan Möbius's avatar
Jan Möbius committed
65

66
  if ( OpenFlipper::Options::gui() && !OpenFlipper::Options::loadingSettings() ) {
Jan Möbius's avatar
Jan Möbius committed
67

Jan Möbius's avatar
Jan Möbius committed
68
69
70
71
    unsigned int maxPases = 1;
    ACG::Vec3d bbmin,bbmax;
    ACG::SceneGraph::analyzeSceneGraph(PluginFunctions::getSceneGraphRootNode(),maxPases,bbmin,bbmax);

Jan Möbius's avatar
Jan Möbius committed
72
    for ( unsigned int i = 0 ; i < OpenFlipper::Options::examinerWidgets() ; ++i ) {
73
      // update scene graph (get new bounding box and set projection right, including near and far plane)
74
      PluginFunctions::viewerProperties(i).lockUpdate();
Jan Möbius's avatar
Jan Möbius committed
75
      coreWidget_->examiner_widgets_[i]->sceneGraph(root_node_scenegraph_,maxPases,bbmin,bbmax, _resetTrackBall );
76
      PluginFunctions::viewerProperties(i).unLockUpdate();
Jan Möbius's avatar
Jan Möbius committed
77
78
79
      coreWidget_->examiner_widgets_[i]->updateGL();
    }

Jan Möbius's avatar
 
Jan Möbius committed
80
  }
81

Jan Möbius's avatar
Jan Möbius committed
82
  // Update the draw Modes Menu
Dirk Wilden's avatar
Dirk Wilden committed
83
84
  if ( OpenFlipper::Options::gui() )
    coreWidget_->slotUpdateGlobalDrawMenu();
Jan Möbius's avatar
 
Jan Möbius committed
85
86
87
88
89
90
91
92
}

//========================================================================================
// ===            Open/Add-Empty Functions                    ============================
//========================================================================================



93
void Core::slotGetAllFilters ( QStringList& _list){
94
  /// \todo check why the supported Type is used here!
Jan Möbius's avatar
Jan Möbius committed
95
  // Iterate over all types
96
97
  for (int i=0; i < (int)supportedTypes().size(); i++){
    QString f = supportedTypes()[i].plugin->getLoadFilters();
Jan Möbius's avatar
 
Jan Möbius committed
98
    f = f.section(")",0,0).section("(",1,1).trimmed();
99
    _list << f;
Jan Möbius's avatar
 
Jan Möbius committed
100
101
102
  }
}

103
void Core::commandLineOpen(const char* _filename, bool _asPolyMesh ){
104

Jan Möbius's avatar
Jan Möbius committed
105
  QString file(_filename);
106

107
  // Modify filename to contain full paths if they were given as relative paths
Jan Möbius's avatar
Jan Möbius committed
108
109
110
111
112
  if ( !file.startsWith("/") && !file.contains(":") ) {
    file = QDir::currentPath();
    file += OpenFlipper::Options::dirSeparator();
    file += _filename;
  }
113

Jan Möbius's avatar
Jan Möbius committed
114
  // Add to the open list
115
  commandLineFileNames_.push_back(std::pair< std::string , bool >(file.toStdString(), _asPolyMesh));
Jan Möbius's avatar
 
Jan Möbius committed
116
117
118
}

void Core::commandLineScript(const char* _filename ) {
119
120
121
122
123
124
125
126
127
128
129
130

  QString file(_filename);
  
  // Modify filename to contain full paths if they were given as relative paths
  if ( !file.startsWith("/") && !file.contains(":") ) {
    file = QDir::currentPath();
    file += OpenFlipper::Options::dirSeparator();
    file += _filename;
  }
  
  // Add to the open list
  commandLineScriptNames_.push_back(file.toStdString());
131
132
}

Jan Möbius's avatar
Jan Möbius committed
133
134
void Core::slotExecuteAfterStartup() {

Jan Möbius's avatar
Jan Möbius committed
135

136
  // Update logger
Jan Möbius's avatar
Jan Möbius committed
137
138
  if ( OpenFlipper::Options::gui())
    coreWidget_->updateInSceneLoggerGeometry();
139

Jan Möbius's avatar
Jan Möbius committed
140
141
142
143
  //check if we have scripting support:
  bool scriptingSupport = false;
  slotPluginExists("scripting",scriptingSupport);
  if ( ! scriptingSupport ) {
144
    emit log(LOGERR ,tr("No scripting support available, please check if we load a scripting plugin .. Skipping script execution on startup"));
Jan Möbius's avatar
Jan Möbius committed
145
146
  }

Jan Möbius's avatar
Jan Möbius committed
147
148
149
  // Collect all script files from the scripting subdirectory and execute them if possible.
  // You can use this directory to execute scipts that modify for example modify the ui at
  // every startup.
Jan Möbius's avatar
Jan Möbius committed
150
151
  if ( scriptingSupport ) {

Jan Möbius's avatar
Jan Möbius committed
152
    // Get the files in the directory
Jan Möbius's avatar
Jan Möbius committed
153
154
155
    QDir scriptDir = OpenFlipper::Options::scriptDir();
    QStringList scriptFiles = scriptDir.entryList(QDir::Files,QDir::Name);

Jan Möbius's avatar
Jan Möbius committed
156
    // Execute all files ending with ofs
Jan Möbius's avatar
Jan Möbius committed
157
    for ( int i = 0 ; i  < scriptFiles.size(); ++i )
158
      if ( scriptFiles[i].endsWith("ofs",Qt::CaseInsensitive) )
159
        emit executeFileScript(scriptDir.path() + "/" + scriptFiles[i]);
Jan Möbius's avatar
Jan Möbius committed
160

161
162
163
    // Clear scripting window afterexecuting the coresubdir scripts
    bool ok = false;
    slotCall( "scripting" ,"clearEditor()",ok);
Jan Möbius's avatar
Jan Möbius committed
164
165
  }

166
167
  OpenFlipper::Options::loadingSettings(true);

Jan Möbius's avatar
Jan Möbius committed
168
  // Open all files given at the commandline
169
  for ( uint i = 0 ; i < commandLineFileNames_.size() ; ++i ) {
170
171

    // Skip scripts here as they will be handled by a different function
172
    QString tmp = QString::fromStdString(commandLineFileNames_[i].first);
173
    if ( tmp.endsWith("ofs",Qt::CaseInsensitive) ) {
174
175
176
177
      commandLineScriptNames_.push_back(commandLineFileNames_[i].first);
      continue;
    }

Jan Möbius's avatar
Jan Möbius committed
178
    // If the file was given with the polymesh option, open them as polymeshes.
179
    if (commandLineFileNames_[i].second)
180
181
182
183
      loadObject(DATA_POLY_MESH, QString::fromStdString(commandLineFileNames_[i].first));
    else {
      loadObject(QString::fromStdString(commandLineFileNames_[i].first));
    }
Jan Möbius's avatar
 
Jan Möbius committed
184
185
  }

186
187
  OpenFlipper::Options::loadingSettings(false);

Jan Möbius's avatar
Jan Möbius committed
188
  // If we have scripting support, execute the scripts given at the commandline.
Jan Möbius's avatar
Jan Möbius committed
189
  if ( scriptingSupport )
190
    for ( uint i = 0 ; i < commandLineScriptNames_.size() ; ++i ) {
191
      emit executeFileScript(QString::fromStdString(commandLineScriptNames_[i]));
192
    }
193

Jan Möbius's avatar
Jan Möbius committed
194
195
  // If we don't have a gui and we are not under remote control,
  // exit the application as there would be no way to execute further commands
Dirk Wilden's avatar
Dirk Wilden committed
196
197
  if ( !OpenFlipper::Options::gui() && !OpenFlipper::Options::remoteControl())
    exitApplication();
Jan Möbius's avatar
 
Jan Möbius committed
198
199
}

200

Jan Möbius's avatar
Jan Möbius committed
201
202
203
204
205
206
207
int Core::loadObject ( QString _filename ) {
  /** \todo Check if this function is ok. It should check all plugins for the given files and do not depend
            on Triangle meshes only! 
            Rewrite function to get the plugin only and then open the file. So iterate over all plugins and find the
            matching ones. open it with this plugin.
  */
  
208
  if (_filename.endsWith(".ini",Qt::CaseInsensitive)) {
209
210

    // Load all information from the given ini file
211
    openIniFile(_filename,true,true,true);
212

Jan Möbius's avatar
 
Jan Möbius committed
213
    if ( OpenFlipper::Options::gui() )
214
      coreWidget_->addRecent(_filename, DATA_UNKNOWN);
215

Jan Möbius's avatar
 
Jan Möbius committed
216
    return -2;
217
  } else if (_filename.endsWith(".ofs",Qt::CaseInsensitive)) {
Jan Möbius's avatar
Jan Möbius committed
218
     emit log(LOGINFO ,tr("Starting script execution of %1.").arg( _filename)) ;
Jan Möbius's avatar
 
Jan Möbius committed
219
     emit executeFileScript(_filename);
Dirk Wilden's avatar
Dirk Wilden committed
220
221
222
  } else {
    
    QFileInfo fi(_filename);
223

224
    for (int i=0; i < (int)supportedTypes().size(); i++){
Dirk Wilden's avatar
Dirk Wilden committed
225

226
      QString filters = supportedTypes()[i].plugin->getLoadFilters();
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243

      // Only take the parts inside the brackets
      filters = filters.section("(",1).section(")",0,0);

      // Split into blocks
      QStringList separateFilters = filters.split(" ");

      bool found = false;

      // for all filters associated with this plugin
      for ( int filterId = 0 ; filterId < separateFilters.size(); ++filterId ) {
        separateFilters[filterId] = separateFilters[filterId].trimmed();

        //check extension
        if ( separateFilters[filterId].endsWith( "*." + fi.completeSuffix() , Qt::CaseInsensitive) ) {
          found = true;
          break;
244
        }
245
246
247
248
249
250
251

        if (  separateFilters[filterId].endsWith( "*." + fi.suffix() , Qt::CaseInsensitive) ) {
          found = true;
          emit log(LOGWARN,"Found supported datatype but only the suffix is matched not the complete suffix!");
          break;
        }

Jan Möbius's avatar
Jan Möbius committed
252
      }
Dirk Wilden's avatar
Dirk Wilden committed
253

254
255
256
257
      // continue processing only if found
      if ( ! found )
        continue;

Dirk Wilden's avatar
Dirk Wilden committed
258
259
260
261
262
263
264
      if ( OpenFlipper::Options::gui() ) {
        coreWidget_->statusMessage( tr("Loading %1 ... ").arg(_filename));
        if ( !OpenFlipper::Options::loadingSettings() )
          coreWidget_->setStatus(ApplicationStatus::PROCESSING );
      }

      //load file
265
      int id = supportedTypes()[i].plugin->loadObject(_filename);
Dirk Wilden's avatar
Dirk Wilden committed
266
267

      if ( OpenFlipper::Options::gui() ) {
268
        if ( id != -1 ) {
Dirk Wilden's avatar
Dirk Wilden committed
269
          coreWidget_->statusMessage( tr("Loading %1 ... done").arg(_filename), 4000 );
270
271
272
273
          
          // Get the object to figure out the data type
          BaseObject* object;
          PluginFunctions::getObject(id,object);
274
275
276
277
278
279
280
281
282
283

          // Security check, if object really exists         
          if ( object != 0 ) { 

            // Add to recent files with the given datatype
            if ( OpenFlipper::Options::gui() )
              coreWidget_->addRecent(_filename, object->dataType());
          } else {
            emit log(LOGERR, tr("Unable to add recent as object with id %1 could not be found!").arg(id) );
          }
284
285
          
        } else
Dirk Wilden's avatar
Dirk Wilden committed
286
287
288
289
290
291
292
293
294
295
296
          coreWidget_->statusMessage( tr("Loading %1 ... failed!").arg(_filename), 4000 );

        if ( !OpenFlipper::Options::loadingSettings() )
          coreWidget_->setStatus(ApplicationStatus::READY );
      }

      return id;
    }
  }

  emit log(LOGERR, tr("Unable to load object (type unknown). No suitable plugin found!") );
Jan Möbius's avatar
 
Jan Möbius committed
297
298
299
300
301
302

  return -1;
}

/// Function for loading a given file
int Core::loadObject( DataType _type, QString _filename) {
Jan Möbius's avatar
Jan Möbius committed
303
304
305
  /** \todo this function has to be checked. test for the plugin which can handle 
            the given file and then use it. 
  */
306
  
307
  if (_type == DATA_UNKNOWN)
Jan Möbius's avatar
 
Jan Möbius committed
308
309
    return loadObject(_filename);

Dirk Wilden's avatar
Dirk Wilden committed
310
311
  QFileInfo fi(_filename);
  
312
313
  for (int i=0; i < (int)supportedTypes().size(); i++)
    if (supportedTypes()[i].type & _type || supportedTypes()[i].type == _type) {
Jan Möbius's avatar
 
Jan Möbius committed
314

315
      QString filters = supportedTypes()[i].plugin->getLoadFilters();
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344

      // Only take the parts inside the brackets
      filters = filters.section("(",1).section(")",0,0);

      // Split into blocks
      QStringList separateFilters = filters.split(" ");

      bool found = false;

      // for all filters associated with this plugin
      for ( int filterId = 0 ; filterId < separateFilters.size(); ++filterId ) {
        separateFilters[filterId] = separateFilters[filterId].trimmed();

        //check extension
        if ( separateFilters[filterId].endsWith( "*." + fi.completeSuffix() , Qt::CaseInsensitive) ) {
          found = true;
          break;
        }

        if (  separateFilters[filterId].endsWith( "*." + fi.suffix() , Qt::CaseInsensitive) ) {
          found = true;
          emit log(LOGWARN,"Found supported datatype but only the suffix is matched not the complete suffix!");
          break;
        }

      }

      // continue processing only if found
      if ( ! found )
Dirk Wilden's avatar
Dirk Wilden committed
345
346
        continue;
      
Jan Möbius's avatar
 
Jan Möbius committed
347
      if ( OpenFlipper::Options::gui() ) {
Jan Möbius's avatar
Jan Möbius committed
348
        coreWidget_->statusMessage( tr("Loading %1 ... ").arg(_filename));
349
        if ( !OpenFlipper::Options::loadingSettings() )
Jan Möbius's avatar
 
Jan Möbius committed
350
351
352
          coreWidget_->setStatus(ApplicationStatus::PROCESSING );
      }

Dirk Wilden's avatar
Dirk Wilden committed
353
354
      int id = -1;

Jan Möbius's avatar
 
Jan Möbius committed
355
      //load file
356
357
      if ( checkSlot( supportedTypes()[i].object , "loadObject(QString,DataType)" ) )
        id = supportedTypes()[i].plugin->loadObject(_filename, _type);
Dirk Wilden's avatar
Dirk Wilden committed
358
      else
359
        id = supportedTypes()[i].plugin->loadObject(_filename);
Jan Möbius's avatar
 
Jan Möbius committed
360
361

      if ( OpenFlipper::Options::gui() ) {
362
        if ( id != -1 ) {
Jan Möbius's avatar
Jan Möbius committed
363
          coreWidget_->statusMessage( tr("Loading %1 ... done").arg(_filename), 4000 );
364
365
366
367
368
369
370
371
372
373
          
          // Get the object to figure out the data type
          BaseObject* object;
          PluginFunctions::getObject(id,object);
          
          // Add to recent files with the given datatype
          if ( OpenFlipper::Options::gui() )
            coreWidget_->addRecent(_filename, object->dataType());
          
        } else
Jan Möbius's avatar
Jan Möbius committed
374
          coreWidget_->statusMessage( tr("Loading %1 ... failed!").arg(_filename), 4000 );
Jan Möbius's avatar
 
Jan Möbius committed
375

376
        if ( !OpenFlipper::Options::loadingSettings() )
Jan Möbius's avatar
 
Jan Möbius committed
377
378
379
380
381
          coreWidget_->setStatus(ApplicationStatus::READY );
      }

      return id;
    }
Dirk Wilden's avatar
Dirk Wilden committed
382
383
384
    
  emit log(LOGERR, tr("Unable to load object. No suitable plugin found!") );
    
Jan Möbius's avatar
 
Jan Möbius committed
385
386
387
  return -1; //no plugin found
}

Jan Möbius's avatar
Jan Möbius committed
388

Jan Möbius's avatar
 
Jan Möbius committed
389
int Core::addEmptyObject( DataType _type ) {
390
  // Iterate over all plugins. The first plugin supporting the addEmpty function for the
391
392
  // specified type will be used to create the new object. If adding the object failed,
  // we iterate over the remaining plugins.
393
  
394
  // Iterate over type plugins
395
  for (int i=0; i < (int)supportedDataTypes_.size(); i++)
396
397
398
399
    if ( supportedDataTypes_[i].type & _type ) {
      int retCode = supportedDataTypes_[i].plugin->addEmpty();
      if ( retCode != -1 )
        return retCode;
Jan Möbius's avatar
Jan Möbius committed
400
    }
401
  
402
  return -1; // no plugin found
Jan Möbius's avatar
 
Jan Möbius committed
403
404
405
406
407
408
409
410
}

//========================================================================================
// ===             Open/Add-Empty Slots                       ============================
//========================================================================================

/// Slot for adding an empty object of given DataType
void Core::slotAddEmptyObject( DataType _type , int& _id ) {
411
412
413
  
  _id = addEmptyObject( _type );
  
Jan Möbius's avatar
Jan Möbius committed
414
415
416
  if ( OpenFlipper::Options::doSlotDebugging() ) {
    if ( sender() != 0 ) {
      if ( sender()->metaObject() != 0 ) {
417
        emit log(LOGINFO,"slotAddEmptyObject( " + _type.name() + "," + QString::number(_id) +  tr(" ) called by ") +
Jan Möbius's avatar
Jan Möbius committed
418
419
420
        QString( sender()->metaObject()->className() ) );
      }
    } else {
421
      emit log(LOGINFO,"slotAddEmptyObject( " + _type.name() + ","  + QString::number(_id) +  tr(" ) called by Core") );
Jan Möbius's avatar
Jan Möbius committed
422
423
    }
  }
Jan Möbius's avatar
 
Jan Möbius committed
424
425
}

426
/// Slot creating a copy of an existing object
Jan Möbius's avatar
Jan Möbius committed
427
void Core::slotCopyObject( int _oldId , int& _newId ) {
428

Jan Möbius's avatar
Jan Möbius committed
429
  if ( _oldId == -1 ) {
Jan Möbius's avatar
Jan Möbius committed
430
    emit log(LOGERR,tr("Requested copy for illegal Object id: %1").arg(_oldId) );
Jan Möbius's avatar
Jan Möbius committed
431
432
433
434
435
436
437
438
    _newId = -1;
    return;
  }

  // get the node
  BaseObject* object = objectRoot_->childExists(_oldId);

  if ( !object ) {
Jan Möbius's avatar
Jan Möbius committed
439
    emit log(LOGERR,tr("Requested copy for unknown Object id: %1 ").arg(_oldId) );
Jan Möbius's avatar
Jan Möbius committed
440
441
442
443
444
445
446
447
    _newId = -1;
    return ;
  }

  // Copy the item
  BaseObject* copy = object->copy();

  if ( copy == 0 ) {
448
    emit log(LOGERR,tr("Unable to create a copy of the object."));
Jan Möbius's avatar
Jan Möbius committed
449
450
451
452
453
454
    return;
  }

  // Integrate into object tree
  copy->setParent( object->parent() );

455
  // return the new id
Jan Möbius's avatar
Jan Möbius committed
456
  _newId = copy->id();
457

458
459
460
461
462
463
  // tell plugins that a new object has been created
  slotEmptyObjectAdded(_newId);

  // tell plugins that the object has been updated
  slotObjectUpdated(_newId);

464
465
}

Dirk Wilden's avatar
Dirk Wilden committed
466
467
468
469
470
471
472
473
474
475
/// Function for loading a given file
void Core::slotLoad(QString _filename, int _pluginID) {

  if ( OpenFlipper::Options::gui() ) {
    coreWidget_->statusMessage( tr("Loading %1 ... ").arg(_filename));
    if ( !OpenFlipper::Options::loadingSettings() )
      coreWidget_->setStatus(ApplicationStatus::PROCESSING );
  }

  //load file
476
  int id = supportedTypes()[_pluginID].plugin->loadObject(_filename);
Dirk Wilden's avatar
Dirk Wilden committed
477
478
479
480
481
482
483
484
485
486
487

  if ( OpenFlipper::Options::gui() ) {
    if ( id != -1 )
      coreWidget_->statusMessage( tr("Loading %1 ... done").arg(_filename), 4000 );
    else
      coreWidget_->statusMessage( tr("Loading %1 ... failed!").arg(_filename), 4000 );

    if ( !OpenFlipper::Options::loadingSettings() )
      coreWidget_->setStatus(ApplicationStatus::READY );
  }
  
488
489
490
491
492
  // Initialize as unknown type
  DataType type = DATA_UNKNOWN;

  // An object has been added. Get it and do some processing!
  if ( id > 0 ) {
493

494
495
    BaseObjectData* object;
    PluginFunctions::getObject(id,object);
Dirk Wilden's avatar
Dirk Wilden committed
496
    
497
    if ( !object ) {
Dirk Wilden's avatar
Dirk Wilden committed
498
499
500
501
502
503
504
505
506
507
508

      BaseObject* baseObj = 0;
      GroupObject* group = 0;
      
      PluginFunctions::getObject(id,baseObj);
      
      if (baseObj){

        group = dynamic_cast< GroupObject* > (baseObj);

        if (group)
509
          type = DATA_GROUP;
Dirk Wilden's avatar
Dirk Wilden committed
510
511
512
513
514
515
      }
      
      if ( group == 0 ){
        emit log(LOGERR,tr("Object id returned but no object with this id has been found! Error in one of the file plugins!"));
        return;
      }
516
517
518
    }
    
    // Get the objects type
Dirk Wilden's avatar
Dirk Wilden committed
519
520
    if (object)
      type = object->dataType();
Dirk Wilden's avatar
Dirk Wilden committed
521
  }
522
523
  
  // If the id was greater than zero, add the file to the recent files.
Dirk Wilden's avatar
Dirk Wilden committed
524
525
526
527
528
  if ( id >= 0 )
    if ( OpenFlipper::Options::gui() )
      coreWidget_->addRecent(_filename, type);
}

Jan Möbius's avatar
 
Jan Möbius committed
529
530
531
/// Slot for loading a given file
void Core::slotLoad(QString _filename, DataType _type, int& _id) {
  _id = loadObject(_type,_filename);
532

Jan Möbius's avatar
 
Jan Möbius committed
533
534
535
  if ( _id < 0 )
    _id = -1;
  else
536
    if ( OpenFlipper::Options::gui() )
Jan Möbius's avatar
 
Jan Möbius committed
537
538
539
540
      coreWidget_->addRecent(_filename,_type);
}

/// Slot gets called after a file-plugin has opened an object
541
void Core::slotFileOpened ( int _id ) {
542
543
544
  if ( OpenFlipper::Options::doSlotDebugging() ) {
    if ( sender() != 0 ) {
      if ( sender()->metaObject() != 0 ) {
545
        emit log(LOGINFO,tr("slotObjectOpened( ") + QString::number(_id) + tr(" ) called by ") +
546
547
548
                  QString( sender()->metaObject()->className() ) );
      }
    } else {
549
      emit log(LOGINFO,tr("slotObjectOpened( ") + QString::number(_id) + tr(" ) called by Core") );
550
551
    }
  }
552
553

  // get the opened object
554
555
  BaseObjectData* object;
  PluginFunctions::getObject(_id,object);
556

Jan Möbius's avatar
Jan Möbius committed
557
558
  // ================================================================================
  // Recompute bounding box and scenegraph info
559
  // Reset scene center here to include new object
Jan Möbius's avatar
Jan Möbius committed
560
  // ================================================================================
561
  resetScenegraph(true);
Jan Möbius's avatar
 
Jan Möbius committed
562

Jan Möbius's avatar
Jan Möbius committed
563
564
565
  // ================================================================================
  // Tell plugins, that a file has been opened
  // ================================================================================
566
  emit openedFile( _id );
567
568
569
570
571
  
  // ================================================================================
  // Print Info to logger
  // ================================================================================
  emit log( LOGINFO ,object->getObjectinfo() );
Jan Möbius's avatar
 
Jan Möbius committed
572

Jan Möbius's avatar
Jan Möbius committed
573
574
575
  // ================================================================================
  // Tell plugins, that the Object is updated and the active object has changed
  // ================================================================================
Jan Möbius's avatar
Jan Möbius committed
576
577
  emit signalObjectUpdated(_id );
  emit signalObjectUpdated(_id, UPDATE_ALL);
Jan Möbius's avatar
 
Jan Möbius committed
578

Jan Möbius's avatar
Jan Möbius committed
579
580
581
  // ================================================================================
  // Create initial backup
  // ================================================================================
582
  emit createBackup(_id,"Original Object");
Jan Möbius's avatar
 
Jan Möbius committed
583

584
585
586
  // ================================================================================
  // Add the file to the recent files menu
  // ================================================================================
587
588
589
  QString filename = object->path() + OpenFlipper::Options::dirSeparator() + object->name();
  BaseObject* object2;
  PluginFunctions::getObject(_id,object2);
Jan Möbius's avatar
 
Jan Möbius committed
590

Jan Möbius's avatar
Jan Möbius committed
591
592
593
  // ================================================================================
  // if this is the first object opend, reset the global view
  // ================================================================================
594
  if ( PluginFunctions::objectCount() == 1 && OpenFlipper::Options::gui() && !OpenFlipper::Options::loadingSettings() )
595
596
597
    for ( unsigned int i = 0 ; i < OpenFlipper::Options::examinerWidgets() ; ++i ) {
      coreWidget_->examiner_widgets_[i]->viewAll();
    }
598

Jan Möbius's avatar
 
Jan Möbius committed
599
  // objectRoot_->dumpTree();
600
}
Jan Möbius's avatar
 
Jan Möbius committed
601
602

 /// Slot gets called after a file-plugin has opened an object
603
void Core::slotEmptyObjectAdded ( int _id ) {
604

605
606
607
  if ( OpenFlipper::Options::doSlotDebugging() ) {
    if ( sender() != 0 ) {
      if ( sender()->metaObject() != 0 ) {
608
        emit log(LOGINFO,tr("slotEmptyObjectAdded( ") + QString::number(_id) + tr(" ) called by ") +
609
610
611
                  QString( sender()->metaObject()->className() ) );
      }
    } else {
612
      emit log(LOGINFO,tr("slotEmptyObjectAdded( ") + QString::number(_id) + tr(" ) called by Core") );
613
614
615
    }
  }

616
617
618
619
  // get the opened object
  BaseObjectData* object;
  PluginFunctions::getObject(_id,object);

620
  
621
  emit emptyObjectAdded( _id );
622

Jan Möbius's avatar
 
Jan Möbius committed
623
  // Tell the Plugins that the Object List and the active object have changed
624
  emit signalObjectUpdated(_id);
Jan Möbius's avatar
Jan Möbius committed
625
  emit signalObjectUpdated(_id,UPDATE_ALL);
Jan Möbius's avatar
 
Jan Möbius committed
626

627
628
  resetScenegraph(false);

629
  ///@todo : set a default path for new objects
Jan Möbius's avatar
 
Jan Möbius committed
630
631
632
//    QString filename = object->path() + OpenFlipper::Options::dirSeparator() + object->name();

//    addRecent(filename);
633
}
Jan Möbius's avatar
 
Jan Möbius committed
634
635
636
637
638
639
640

//========================================================================================
// ===             Menu Slots                                 ============================
//========================================================================================

/// Opens AddEmpty-Object widget
void Core::slotAddEmptyObjectMenu() {
641
642
643
644
645
646
647
648
649
650
651
652
653
  std::vector< DataType > types;
  QStringList             typeNames;
  
  DataType currentType = 2;
  
  // Iterate over all Types known to the core
  // Start at 2:
  // 0 type is defined as DATA_UNKNOWN
  // 1 type is defined as DATA_GROUP
  // Therefore we have two types less then reported
  // 
  for ( uint i = 0 ; i < typeCount() - 2  ; ++i) {
    
654
    // Iterate over all supported types (created from plugins on load)
655
656
    // Check if a plugin supports addEmpty for the current type.
    // Only if the type is supported, add it to the addEmpty Dialog
657
658
659
    
    // typePlugin
    for ( uint j = 0 ; j < supportedDataTypes_.size(); j++) {
660
      
661
662
      // Check if a plugin supports the current type
      if ( supportedDataTypes_[j].type & currentType ) {
663
664
665
666
667
        types.push_back(currentType);
        typeNames.push_back( typeName( currentType ) );
        
        // Stop here as we need only one plugin supporting addEmpty for a given type
        break;
Jan Möbius's avatar
   
Jan Möbius committed
668
      }
669
670
    }
    
671
    // filePlugin
672
    for ( uint j = 0 ; j < supportedTypes().size(); j++) {
673
674
      
      // Check if a plugin supports the current Type
675
      if ( supportedTypes()[j].type & currentType ) {
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
	
	// Avoid duplicates
	bool duplicate = false;
	for(std::vector< DataType >::iterator it = types.begin(); it != types.end(); ++it) {
	  if(*it == currentType) {
	    duplicate = true;
	    break;
	  }
	}
	  
	if(!duplicate) {
	  types.push_back(currentType);
	  typeNames.push_back( typeName( currentType ) );
        
	  // Stop here as we need only one plugin supporting addEmpty for a given type
	  break;
	}
      }
    }
    
696
697
    // Advance to next type ( Indices are bits so multiply by two to get next bit)
    ++currentType;
698
699
700
701
  }
  
  static addEmptyWidget* widget = 0;
  
702
  if (supportedTypes().size() != 0) {
703
704
    
    if ( !widget ){
Jan Möbius's avatar
 
Jan Möbius committed
705
      widget = new addEmptyWidget(types,typeNames);
706
      widget->setWindowIcon( OpenFlipper::Options::OpenFlipperIcon() );
Jan Möbius's avatar
 
Jan Möbius committed
707
708
      connect(widget,SIGNAL(chosen(DataType, int&)),this,SLOT(slotAddEmptyObject(DataType, int&)));
    }
709
    
Jan Möbius's avatar
 
Jan Möbius committed
710
    widget->show();
711
712
713
714
    
  } else
    emit log(LOGERR,tr("Could not show 'add Empty' dialog. Missing file-plugins ?"));
  
Jan Möbius's avatar
 
Jan Möbius committed
715
716
}

717
718
719
720
//========================================================================================
// ===             Public Slots                                 ============================
//========================================================================================

Jan Möbius's avatar
 
Jan Möbius committed
721
/// Open Load-Object Widget
722
723
724
725
void Core::loadObject() {

  if ( OpenFlipper::Options::gui() ){

726
727
    if (supportedTypes().size() != 0){
      LoadWidget* widget = new LoadWidget(supportedTypes());
Dirk Wilden's avatar
Dirk Wilden committed
728
729
      connect(widget,SIGNAL(load(QString, int)),this,SLOT(slotLoad(QString, int)));
      connect(widget,SIGNAL(save(int, QString, int)),this,SLOT(saveObject(int, QString, int)));
Jan Möbius's avatar
Jan Möbius committed
730

731
732
      widget->setWindowIcon( OpenFlipper::Options::OpenFlipperIcon() );

733
      widget->showLoad();
734
735
736
737

      widget->disconnect();
      delete widget;

738
    }else
739
      emit log(LOGERR,tr("Could not show 'load objects' dialog. Missing file-plugins."));
Jan Möbius's avatar
 
Jan Möbius committed
740

741
  }
Jan Möbius's avatar
 
Jan Möbius committed
742
743
}

744
745
746
747
748
749
/// Load settings from file
void Core::loadSettings(){

  if ( OpenFlipper::Options::gui() ){

    QString complete_name;
Jan Möbius's avatar
Jan Möbius committed
750

751
752
753

    QFileDialog fileDialog( coreWidget_,
                            tr("Load Settings"),
754
                            OpenFlipperSettings().value("Core/CurrentDir").toString(),
755
                            tr("INI files (*.ini)") );
756

757
    fileDialog.setOption (QFileDialog::DontUseNativeDialog, true);
758
759
760
761
762
763
764
    fileDialog.setAcceptMode ( QFileDialog::AcceptOpen );
    fileDialog.setFileMode ( QFileDialog::AnyFile );

    QGridLayout *layout = (QGridLayout*)fileDialog.layout();

    QGroupBox* optionsBox = new QGroupBox( &fileDialog ) ;
    optionsBox->setSizePolicy( QSizePolicy ( QSizePolicy::Expanding , QSizePolicy::Preferred ) );
765
    optionsBox->setTitle(tr("Options"));
766
767
768
    layout->addWidget( optionsBox, layout->rowCount() , 0 , 1,layout->columnCount() );

    QCheckBox *loadProgramSettings = new QCheckBox(optionsBox);
769
770
    loadProgramSettings->setText(tr("Load program settings"));
    loadProgramSettings->setToolTip(tr("Load all current program settings from the file ( This will include view settings, colors,...) "));
771
772
773
    loadProgramSettings->setCheckState( Qt::Unchecked );

    QCheckBox *loadPluginSettings = new QCheckBox(optionsBox);
774
775
    loadPluginSettings->setText(tr("Load per Plugin Settings"));
    loadPluginSettings->setToolTip(tr("Plugins should load their current global settings from the file"));
776
777
778
    loadPluginSettings->setCheckState( Qt::Checked );

    QCheckBox *loadObjectInfo = new QCheckBox(optionsBox);
779
780
    loadObjectInfo->setText(tr("Load all objects defined in the file"));
    loadObjectInfo->setToolTip(tr("Load all objects which are defined in the file"));
781
782
783
784
785
786
787
788
    loadObjectInfo->setCheckState( Qt::Checked );

    QBoxLayout* frameLayout = new QBoxLayout(QBoxLayout::TopToBottom,optionsBox);
    frameLayout->addWidget( loadProgramSettings , 0 , 0);
    frameLayout->addWidget( loadPluginSettings  , 1 , 0);
    frameLayout->addWidget( loadObjectInfo      , 2 , 0);
    frameLayout->addStretch();

Jan Möbius's avatar
Jan Möbius committed
789
790
    fileDialog.resize(550 ,500);

791
792
793
794
795
796
797
798
799
800
801
802
    // ========================================================================================
    // show the saveSettings-Dialog and get the target file
    // ========================================================================================
    QStringList fileNames;
    if (fileDialog.exec()) {
      fileNames = fileDialog.selectedFiles();
    } else {
      return;
    }

    if ( fileNames.size() > 1 ) {
      std::cerr << "Too many save filenames selected" << std::endl;
803
      return;
804
805
806
807
    }

    complete_name = fileNames[0];

Jan Möbius's avatar
Jan Möbius committed
808

809
    QString newpath = complete_name.section(OpenFlipper::Options::dirSeparator(),0,-2);
810
    OpenFlipperSettings().setValue("Core/CurrentDir", newpath);
Jan Möbius's avatar
Jan Möbius committed
811

812
    if ( complete_name.endsWith("ini",Qt::CaseInsensitive) ) {
813
814
815
816
      openIniFile( complete_name,
                   loadProgramSettings->isChecked(),
                   loadPluginSettings->isChecked(),
                   loadObjectInfo->isChecked());
817
818
      if ( loadProgramSettings->isChecked() )
        applyOptions();
819
820
    } 
    
821
    coreWidget_->addRecent(complete_name, DATA_UNKNOWN);
822
823
  }
}
Jan Möbius's avatar
 
Jan Möbius committed
824

825
826
/// Load settings from file
void Core::loadSettings(QString _filename){
Jan Möbius's avatar
Jan Möbius committed
827

828
  if ( !QFile(_filename).exists() )
Jan Möbius's avatar
 
Jan Möbius committed
829
830
    return;

831
  QString newpath = _filename.section(OpenFlipper::Options::dirSeparator(),0,-2);
832
  OpenFlipperSettings().setValue("Core/CurrentDir", newpath);
Jan Möbius's avatar
 
Jan Möbius committed
833

834
  if ( _filename.endsWith("ini",Qt::CaseInsensitive) ) {
835
    // Loaded function for recent files. Load everything.
836
    openIniFile(_filename,true,true,true);
Jan Möbius's avatar
 
Jan Möbius committed
837
    applyOptions();
838
  } else if ( _filename.endsWith("obj",Qt::CaseInsensitive) ) {
Dirk Wilden's avatar
Dirk Wilden committed
839
    loadObject(_filename);
Jan Möbius's avatar
 
Jan Möbius committed
840
841
842
843
    applyOptions();
  }

}