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();
    }

80
    coreWidget_->slotUpdateGlobalDrawMenu();
Jan Möbius's avatar
 
Jan Möbius committed
81
  }
82

Jan Möbius's avatar
 
Jan Möbius committed
83
84
85
86
87
88
89
90
}

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



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

101
void Core::commandLineOpen(const char* _filename, bool _asPolyMesh ){
102

Jan Möbius's avatar
Jan Möbius committed
103
  QString file(_filename);
104

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

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

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

  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());
129
130
}

Jan Möbius's avatar
Jan Möbius committed
131
132
void Core::slotExecuteAfterStartup() {

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

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

Jan Möbius's avatar
Jan Möbius committed
138
139
140
141
  //check if we have scripting support:
  bool scriptingSupport = false;
  slotPluginExists("scripting",scriptingSupport);
  if ( ! scriptingSupport ) {
142
    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
143
144
  }

Jan Möbius's avatar
Jan Möbius committed
145
146
147
  // 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
148
149
  if ( scriptingSupport ) {

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

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

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

164
165
  OpenFlipper::Options::loadingSettings(true);

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

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

Jan Möbius's avatar
Jan Möbius committed
176
    // If the file was given with the polymesh option, open them as polymeshes.
177
    if (commandLineFileNames_[i].second)
178
179
180
181
      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
182
183
  }

184
185
  OpenFlipper::Options::loadingSettings(false);

186
187
188
  // Reset the scenegraph once to make sure everything is fine
  resetScenegraph( true );

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

Jan Möbius's avatar
Jan Möbius committed
195
196
  // 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
197
198
  if ( !OpenFlipper::Options::gui() && !OpenFlipper::Options::remoteControl())
    exitApplication();
Jan Möbius's avatar
 
Jan Möbius committed
199
200
}

201

Jan Möbius's avatar
Jan Möbius committed
202
203
204
205
206
207
208
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.
  */
  
209
  if (_filename.endsWith(".ini",Qt::CaseInsensitive)) {
210
211

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

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

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

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

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

      // 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;
245
        }
246
247
248
249
250
251
252

        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
253
      }
Dirk Wilden's avatar
Dirk Wilden committed
254

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

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

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

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

          // 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) );
          }
285
286
          
        } else
Dirk Wilden's avatar
Dirk Wilden committed
287
288
289
290
291
292
293
294
295
296
297
          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
298
299
300
301
302
303

  return -1;
}

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

Dirk Wilden's avatar
Dirk Wilden committed
311
312
  QFileInfo fi(_filename);
  
313
314
  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
315

316
      QString filters = supportedTypes()[i].plugin->getLoadFilters();
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
345

      // 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
346
347
        continue;
      
Jan Möbius's avatar
 
Jan Möbius committed
348
      if ( OpenFlipper::Options::gui() ) {
Jan Möbius's avatar
Jan Möbius committed
349
        coreWidget_->statusMessage( tr("Loading %1 ... ").arg(_filename));
350
        if ( !OpenFlipper::Options::loadingSettings() )
Jan Möbius's avatar
 
Jan Möbius committed
351
352
353
          coreWidget_->setStatus(ApplicationStatus::PROCESSING );
      }

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

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

      if ( OpenFlipper::Options::gui() ) {
363
        if ( id != -1 ) {
Jan Möbius's avatar
Jan Möbius committed
364
          coreWidget_->statusMessage( tr("Loading %1 ... done").arg(_filename), 4000 );
365
366
367
368
369
370
371
372
373
374
          
          // 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
375
          coreWidget_->statusMessage( tr("Loading %1 ... failed!").arg(_filename), 4000 );
Jan Möbius's avatar
 
Jan Möbius committed
376

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

465
466
}

Dirk Wilden's avatar
Dirk Wilden committed
467
468
469
470
471
472
473
474
475
476
/// 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
477
  int id = supportedTypes()[_pluginID].plugin->loadObject(_filename);
Dirk Wilden's avatar
Dirk Wilden committed
478
479
480
481
482
483
484
485
486
487
488

  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 );
  }
  
489
490
491
492
493
  // Initialize as unknown type
  DataType type = DATA_UNKNOWN;

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

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

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

        group = dynamic_cast< GroupObject* > (baseObj);

        if (group)
510
          type = DATA_GROUP;
Dirk Wilden's avatar
Dirk Wilden committed
511
512
513
514
515
516
      }
      
      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;
      }
517
518
519
    }
    
    // Get the objects type
Dirk Wilden's avatar
Dirk Wilden committed
520
521
    if (object)
      type = object->dataType();
Dirk Wilden's avatar
Dirk Wilden committed
522
  }
523
524
  
  // If the id was greater than zero, add the file to the recent files.
Dirk Wilden's avatar
Dirk Wilden committed
525
526
527
528
529
  if ( id >= 0 )
    if ( OpenFlipper::Options::gui() )
      coreWidget_->addRecent(_filename, type);
}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

621
  
622
  emit emptyObjectAdded( _id );
623

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

628
629
  resetScenegraph(false);

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

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

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

/// Opens AddEmpty-Object widget
void Core::slotAddEmptyObjectMenu() {
642
643
644
645
646
647
648
649
650
651
652
653
654
  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) {
    
655
    // Iterate over all supported types (created from plugins on load)
656
657
    // Check if a plugin supports addEmpty for the current type.
    // Only if the type is supported, add it to the addEmpty Dialog
658
659
660
    
    // typePlugin
    for ( uint j = 0 ; j < supportedDataTypes_.size(); j++) {
661
      
662
663
      // Check if a plugin supports the current type
      if ( supportedDataTypes_[j].type & currentType ) {
664
665
666
667
668
        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
669
      }
670
671
    }
    
672
    // filePlugin
673
    for ( uint j = 0 ; j < supportedTypes().size(); j++) {
674
675
      
      // Check if a plugin supports the current Type
676
      if ( supportedTypes()[j].type & currentType ) {
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
	
	// 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;
	}
      }
    }
    
697
698
    // Advance to next type ( Indices are bits so multiply by two to get next bit)
    ++currentType;
699
700
701
702
  }
  
  static addEmptyWidget* widget = 0;
  
703
  if (supportedTypes().size() != 0) {
704
705
    
    if ( !widget ){
Jan Möbius's avatar
 
Jan Möbius committed
706
      widget = new addEmptyWidget(types,typeNames);
707
      widget->setWindowIcon( OpenFlipper::Options::OpenFlipperIcon() );
Jan Möbius's avatar
 
Jan Möbius committed
708
709
      connect(widget,SIGNAL(chosen(DataType, int&)),this,SLOT(slotAddEmptyObject(DataType, int&)));
    }
710
    
Jan Möbius's avatar
 
Jan Möbius committed
711
    widget->show();
712
713
714
715
    
  } else
    emit log(LOGERR,tr("Could not show 'add Empty' dialog. Missing file-plugins ?"));
  
Jan Möbius's avatar
 
Jan Möbius committed
716
717
}

718
719
720
721
//========================================================================================
// ===             Public Slots                                 ============================
//========================================================================================

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

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

727
728
    if (supportedTypes().size() != 0){
      LoadWidget* widget = new LoadWidget(supportedTypes());
Dirk Wilden's avatar
Dirk Wilden committed
729
730
      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
731

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

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

      widget->disconnect();
      delete widget;

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

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

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

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

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

752
753
754

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

758
    fileDialog.setOption (QFileDialog::DontUseNativeDialog, true);
759
760
761
762
763
764
765
    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 ) );
766
    optionsBox->setTitle(tr("Options"));
767
768
769
    layout->addWidget( optionsBox, layout->rowCount() , 0 , 1,layout->columnCount() );

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

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

    QCheckBox *loadObjectInfo = new QCheckBox(optionsBox);
780
781
    loadObjectInfo->setText(tr("Load all objects defined in the file"));
    loadObjectInfo->setToolTip(tr("Load all objects which are defined in the file"));
782
783
784
785
786
787
788
789
    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
790
791
    fileDialog.resize(550 ,500);

792
793
794
795
796
797
798
799
800
801
802
803
    // ========================================================================================
    // 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;
804
      return;
805
806
807
808
    }

    complete_name = fileNames[0];

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

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

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

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

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

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

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

}