view libgui/src/workspace-model.cc @ 15896:57be060d7672 classdef

Implement full object construction and superclass references. * libinterp/octave-value/ov-classdef.h (cdef_object_rep::subsref): Add class context argument. (cdef_object_rep::ctor_list): New class member. (cdef_object_rep::mark_for_construction, cdef_object_rep::is_constructed_for, cdef_object_rep::is_partially_constructed_for, cdef_object_rep::mark_as_constructed, cdef_object_rep::is_constructed, cdef_object::mark_for_construction, cdef_object::is_constructed_for, cdef_object::is_partially_constructed_for, cdef_object::mark_as_constructed, cdef_object:is_constructed): New methods to manipulate the new ctor_list class member. (cdef_object_rep::cdef_object_rep (const cdef_object_rep&)): New copy constructor. (cdef_class::cdef_class_rep (std::string, std::list<cdef_class>)): New utility constructor. (cdef_class (std::string, std::list<cdef_class>)): Likewise. (cdef_class::cdef_class_rep::implicit_ctor_list): New class member. (cdef_method::cdef_method_rep::is_constructor, cdef_method::is_construcror): New methods. (cdef_method::get_function): Likewise. (cdef_package::cdef_package_rep::subsref): Remove method. (octave_classdef::get_object_ref): New method. (to_cdef_ref): New function. * libinterp/octave-value/ov-classdef.cc (is_superclass): Add max_depth argument. (is_direct_superclass): New utility function. (get_class_context): Add bool and std::string return values, as new reference arguments. Add overload without any arguments. (check_access): Check for valid class context when required. (make_class): Remove super_name_list argument. Do not set SuperClasses explicitly, use new cdef_class constructor instead. (octave_classdef::subsref): Change skip variable to size_t (avoids compilation warning). Call cdef_object::subsref with additional class context argument. (class octave_classdef_superclass_ref): New utility class. (F__superclass_reference__): Use it. (cdef_object_rep::subsref): Add class context argument. (cdef_object_rep::mark_for_construction): New method. (handle_cdef_object::~handle_cdef_object, value_cdef_object::~value_cdef_object, cdef_class::cdef_class_rep::subsref_meta, cdef_class::make_meta_class): Use gnulib::printf explicitly. (cdef_class::cdef_class_rep (std::string, std::list<cdef_class>)): New constructor. (class ctor_analyzer): New utility class. (cdef_class::cdef_class_rep::install_method): Use it to analyze constructors and detect explicit superclass constructor calls. (cdef_class::cdef_class_rep::initialize_object): Call cdef_object::mark_for_construction. (cdef_class::cdef_class_rep::run_constructor): Run all implicit superclass constructors first. (cdef_class::cdef_class_rep::construct): Call cdef_object::mark_as_constructed. (cdef_class::make_meta_class): Remove snamelist variable. (cdef_property::cdef_property_rep::get_value, cdef_property::cdef_property_rep::set_value): Check if object is partially constructed for the property-defining class. (cdef_method::cdef_method_rep::is_constructor): New method. (cdef_package::cdef_package_rep::subsref): Delete method.
author Michael Goffioul <michael.goffioul@gmail.com>
date Thu, 03 Jan 2013 21:01:11 -0500
parents 7f423c6111c6
children 0486a29d780f
line wrap: on
line source

/*

Copyright (C) 2011-2012 Jacob Dawid

This file is part of Octave.

Octave is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation; either version 3 of the License, or (at your
option) any later version.

Octave 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 General Public License
for more details.

You should have received a copy of the GNU General Public License
along with Octave; see the file COPYING.  If not, see
<http://www.gnu.org/licenses/>.

*/

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <QTreeWidget>
#include <QTime>

#include <list>

#include "symtab.h"
#include "variables.h"

#include "workspace-model.h"
#include "octave-link.h"

workspace_model::workspace_model(QObject *p)
  : QAbstractItemModel (p)
{
  QList<QVariant> rootData;
  rootData << tr ("Name") << tr ("Class") << tr("Dimension") << tr ("Value");
  _rootItem = new tree_item(rootData);

  insert_top_level_item(0, new tree_item ("Local"));
  insert_top_level_item(1, new tree_item ("Global"));
  insert_top_level_item(2, new tree_item ("Persistent"));

  connect(&_update_workspace_model_timer,
          SIGNAL (timeout ()),
          this,
          SLOT (request_update_workspace()));

  _update_workspace_model_timer.setInterval (500);
  _update_workspace_model_timer.setSingleShot (true);
  _update_workspace_model_timer.start ();
}

workspace_model::~workspace_model()
{
  delete _rootItem;
}

void
workspace_model::request_update_workspace ()
{
  octave_link::post_event (this, &workspace_model::update_workspace_callback);
}

QModelIndex
workspace_model::index(int row, int column, const QModelIndex &p) const
{
  if (!hasIndex(row, column, p))
    return QModelIndex();

  tree_item *parentItem;

  if (!p.isValid())
    parentItem = _rootItem;
  else
    parentItem = static_cast<tree_item*>(p.internalPointer());

  tree_item *childItem = parentItem->child(row);
  if (childItem)
    return createIndex(row, column, childItem);
  else
    return QModelIndex();
}

QModelIndex
workspace_model::parent(const QModelIndex &idx) const
{
  if (!idx.isValid())
    return QModelIndex();

  tree_item *childItem = static_cast<tree_item*>(idx.internalPointer());

  if (childItem)
    {
      tree_item *parentItem = childItem->parent();

      if (! parentItem || parentItem == _rootItem)
        return QModelIndex();

      return createIndex(parentItem->row(), 0, parentItem);
    }
  else
    return QModelIndex ();
}

int
workspace_model::rowCount(const QModelIndex &p) const
{
  tree_item *parentItem;
  if (p.column() > 0)
    return 0;

  if (!p.isValid())
    parentItem = _rootItem;
  else
    parentItem = static_cast<tree_item*>(p.internalPointer());

  return parentItem->child_count();
}

int
workspace_model::columnCount(const QModelIndex &p) const
{
  if (p.isValid())
    return static_cast<tree_item*>(p.internalPointer())->column_count();
  else
    return _rootItem->column_count();
}

void
workspace_model::insert_top_level_item(int at, tree_item *treeItem)
{
  _rootItem->insert_child_item(at, treeItem);
}

tree_item *
workspace_model::top_level_item (int at)
{
  return _rootItem->child(at);
}

Qt::ItemFlags
workspace_model::flags(const QModelIndex &idx) const
{
  if (!idx.isValid())
    return 0;

  return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
}

QVariant
workspace_model::headerData(int section, Qt::Orientation orientation, int role) const
{
  if (orientation == Qt::Horizontal && role == Qt::DisplayRole)
    return _rootItem->data(section);

  return QVariant();
}

QVariant
workspace_model::data(const QModelIndex &idx, int role) const
{
  if (!idx.isValid())
    return QVariant();

  if (role != Qt::DisplayRole)
    return QVariant();

  tree_item *item = static_cast<tree_item*>(idx.internalPointer());

  return item->data(idx.column());
}

void
workspace_model::update_workspace_callback (void)
{
  std::list < symbol_table::symbol_record > symbolTable = symbol_table::all_variables ();

  _symbol_information.clear ();
  for (std::list < symbol_table::symbol_record > ::iterator iterator = symbolTable.begin ();
       iterator != symbolTable.end (); iterator++)
    _symbol_information.push_back (symbol_information (*iterator));

  beginResetModel();
  top_level_item (0)->delete_child_items ();
  top_level_item (1)->delete_child_items ();
  top_level_item (2)->delete_child_items ();

  foreach (const symbol_information& s, _symbol_information)
    {
      tree_item *child = new tree_item ();

      child->set_data (0, s.symbol ());
      child->set_data (1, s.class_name ());
      child->set_data (2, s.dimension ());
      child->set_data (3, s.value ());

      switch (s.scope ())
        {
        case symbol_information::local:
          top_level_item (0)->add_child (child);
          break;

        case symbol_information::global:
          top_level_item (1)->add_child (child);
          break;

        case symbol_information::persistent:
          top_level_item (2)->add_child (child);
          break;

        default:
          break;
        }
    }

  endResetModel();
  emit model_changed();

  // Post a new event in a given time.
  // This prevents flooding the event queue when no events are being processed.
  _update_workspace_model_timer.start ();
}