// This may look like C code, but it's really -*- C++ -*-
/*
 * Copyright (C) 2008 Emweb bvba, Kessel-Lo, Belgium.
 *
 * See the LICENSE file for terms of use.
 */
#ifndef CHART_WABSTRACT_CHART_H_
#define CHART_WABSTRACT_CHART_H_

#include <Wt/WBrush>
#include <Wt/WFont>
#include <Wt/WPaintedWidget>
#include <Wt/WRectF>

namespace Wt {

  class WAbstractItemModel;
  class WModelIndex;
  class WPainter;
  class WRectF;

  /*! \brief Namespace for the \ref charts
   */
  namespace Chart {

    class WChartPalette;

/*! \defgroup charts Charts (Wt::Chart)
 *  \brief A charting library implemented using the %Wt \ref painting
 *
 * The charting library contains two main chart widget classes,
 * WCartesianChart and WPieChart, and a number of utility classes
 * for drawing simple to complex charts.
 */

/*! \class WAbstractChart Wt/Chart/WAbstractChart Wt/Chart/WAbstractChart
 *  \brief Abstract base class for MVC-based charts.
 *
 * This is an abstract class and should not be used directly.
 *
 * As an abstract base for MVC-based charts, this class manages the
 * model setModel() and provides virtual methods that listen to
 * model changes. In addition, it gives access to generic chart
 * properties such as the title setTitle() and title font
 * setTitleFont(), the chart palette setPalette(), plot area
 * padding setPlotAreaPadding(), and the background fill color
 * setBackground().
 *
 * <h3>CSS</h3>
 *
 * Styling through CSS is not applicable.
 *
 * \sa WCartesianChart, WPieChart
 *
 * \ingroup charts modelview
 */
class WT_API WAbstractChart : public WPaintedWidget
{
public:
  /*! \brief Destructor.
   */
  virtual ~WAbstractChart();

  /*! \brief Set the model.
   *
   * The model is used by the chart to get its data. Ownership of the
   * model is not transferred, and if a previous model was set it is
   * not deleted.
   *
   * The default model is a 0 model.
   *
   * \sa model()
   */
  void setModel(WAbstractItemModel *model);

  /*! \brief Returns the model.
   *
   * \sa setModel(WAbstractItemModel *)
   */
  WAbstractItemModel *model() const { return model_; }

  /*! \brief Sets a background for the chart.
   *
   * Set the background color for the main plot area.
   *
   * The default is a completely transparent background.
   *
   * \sa background()
   */
  void setBackground(const WBrush& background);

  /*! \brief Returns the background of the chart.
   *
   * \sa setBackground(const WBrush&)
   */
  const WBrush& background() const { return background_; }

  /*! \brief Set a palette for the chart.
   *
   * A palette is used to provide the style information to render the
   * chart series. Ownership of the palette is transferred to the chart.
   *
   * The default palette is dependent on the chart type.
   *
   * \sa palette()
   */
  void setPalette(WChartPalette *palette);

  /*! \brief Returns the palette for the chart.
   *
   * \sa setPalette(WChartPalette *palette)
   */
  WChartPalette *palette() const { return palette_; }

  /*! \brief Set an internal margin for the main plot area.
   *
   * This configures the area (in pixels) around the plot area that is
   * available for axes, labels, and titles. You need to set this
   * appropriately so that labels fit inside these margins.
   *
   * The default is dependent on the chart type.
   */
  void setPlotAreaPadding(int padding, WFlags<Side> sides = All);

  /*! \brief Returns the internal margin for the main plot area.
   *
   * \sa setPlotAreaPadding(int, WFlags<Side>)
   */
  int plotAreaPadding(Side side) const;

  /*! \brief Set a chart title.
   *
   * The title is displayed on top of the chart, using the titleFont().
   *
   * The default title is an empty title ("").
   *
   * \sa title()
   */
  void setTitle(const WString& title);

  /*! \brief Return the chart title.
   *
   * \sa title()
   */
  const WString& title() const { return title_; }

  /*! \brief Set the font for the chart title.
   *
   * Changes the font for the chart title.
   *
   * The default title font is a 15 point Sans Serif font.
   *
   * \sa titleFont(), setTitle(const WString&)
   */
  void setTitleFont(const WFont& titleFont);

  /*! \brief Returns the font for the chart title.
   *
   * \sa setTitleFont(const WFont&)
   */
  const WFont& titleFont() const { return titleFont_; }

  void setAxisTitleFont(const WFont& titleFont);
  const WFont& axisTitleFont() const { return titleFont_; }

  /*! \brief Paint the chart in a rectangle of the given painter.
   *
   * Paints the chart inside the <i>painter</i>, in the area indicated
   * by <i>rectangle</i>.  When <i>rectangle</i> is a null rectangle,
   * the entire painter \link WPainter::window() window\endlink is
   * used.
   */
  virtual void paint(WPainter& painter, const WRectF& rectangle = WRectF())
    const = 0;

protected:
  WAbstractChart(WContainerWidget *parent);

private:
  WAbstractItemModel *model_;
  WBrush              background_;
  WChartPalette      *palette_;
  int                 padding_[4];
  WString             title_;
  WFont               titleFont_;
  WFont               axisTitleFont_;

  // connections with the current model, used to disconnect from a model
  // when the model changes.
  std::vector<boost::signals::connection> modelConnections_;

  /*! \brief Method called whenever the entire model was changed.
   *
   * \sa setModel(WAbstractItemModel *)
   */
  virtual void modelChanged();

  /*! \brief Method called whenever the entire model was reset.
   *
   * Bound to the WAbstractItemModel::modelReset() and
   * WAbstractItemModel::layoutChanged() signals.
   */
  virtual void modelReset();

  /*! \brief Method called when colums have been inserted in the model.
   *
   * \sa WAbstractItemModel::columnsInserted
   */
  virtual void modelColumnsInserted(const WModelIndex& parent,
				    int start, int end) = 0;

  /*! \brief Method called when colums have been removed from the model.
   *
   * \sa WAbstractItemModel::columnsRemoved
   */
  virtual void modelColumnsRemoved(const WModelIndex& parent,
				   int start, int end) = 0;

  /*! \brief Method called when rows have been inserted from the model.
   *
   * \sa WAbstractItemModel::rowsInserted
   */
  virtual void modelRowsInserted(const WModelIndex& parent,
				 int start, int end) = 0;

  /*! \brief Method called when rows have been removed from the model.
   *
   * \sa WAbstractItemModel::rowsRemoved
   */
  virtual void modelRowsRemoved(const WModelIndex& parent,
				int start, int end) = 0;

  /*! \brief Method called when data has been changed in the model.
   *
   * \sa WAbstractItemModel::dataChanged
   */
  virtual void modelDataChanged(const WModelIndex& topLeft,
				const WModelIndex& bottomRight) = 0;

  template <typename T>
  void set(T& m, const T& v);
};

template <typename T>
void WAbstractChart::set(T& m, const T& v)
{
  if (m != v) {
    m = v;
    update();
  }
}


  }
}

#endif // CHART_WABSTRACT_CHART_H_
