// Copyright (C) 2010, Guy Barrand. All rights reserved.
// See the file tools.license for terms.

#ifndef tools_store_osc_streamers
#define tools_store_osc_streamers

#include "iobj_const_visitor"
#include "iobj_visitor"

#include "../histo/h1d"
#include "../histo/h2d"
#include "../histo/h3d"
#include "../histo/p1d"
#include "../histo/p2d"
#include "../vmanip"
#include "../sto"
#include "../num2s"
#include "../S_STRING"
#include "../scast"

namespace tools {
namespace osc {

inline const std::string& s_axis() {
  static const std::string s_v("BatchLab::Axis");
  return s_v;
}
inline const std::string& s_annotation() {
  static const std::string s_v("BatchLab::Annotation");
  return s_v;
}
inline const std::string& s_base_histogram() {
  static const std::string s_v("BatchLab::BaseHistogram");
  return s_v;
}
inline const std::string& s_item() {
  static const std::string s_v("BatchLab::Item");
  return s_v;
}

inline const std::string& s_h1d() {
  static const std::string s_v("BatchLab::Histogram1D");
  return s_v;
}

inline const std::string& s_h2d() {
  static const std::string s_v("BatchLab::Histogram2D");
  return s_v;
}

inline const std::string& s_h3d() {
  static const std::string s_v("BatchLab::Histogram3D");
  return s_v;
}

inline const std::string& s_p1d() {
  static const std::string s_v("BatchLab::Profile1D");
  return s_v;
}

inline const std::string& s_p2d() {
  static const std::string s_v("BatchLab::Profile2D");
  return s_v;
}

class Axis : public virtual istorable {
public:
  TOOLS_SCLASS(tools::osc::Axis)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<Axis>(this,a_class)) return p;
    return 0;
  }
  virtual const std::string& store_cls() const {return s_axis();}
  virtual bool visit(iobj_const_visitor& a_v) const {
    if(!a_v.begin(*this,s_axis(),Axis::s_visit)) return false;

    int version = 1;
    if(!a_v.visit("fVersion",version)) return false;

    if(!a_v.visit("fOffset",m_axis.m_offset)) return false;
    if(!a_v.visit("fNumberOfBins",(int)m_axis.m_number_of_bins)) return false;
    if(!a_v.visit("fMinimumValue",m_axis.m_minimum_value)) return false;
    if(!a_v.visit("fMaximumValue",m_axis.m_maximum_value)) return false;
    if(!a_v.visit("fFixed",m_axis.m_fixed)) return false;
    if(!a_v.visit("fBinWidth",m_axis.m_bin_width)) return false;
    if(!a_v.visit("fEdges",m_axis.m_edges)) return false;

    if(!a_v.end(*this)) return false;
    return true;
  }
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const Axis* local = safe_cast<istorable,Axis>(a_o);
    if(!local) return false;
    return local->Axis::visit(a_v); //IMPORTANT : have Axis::
  }
public:
  Axis(const histo::axis<double,unsigned int>& a_axis):m_axis(a_axis){}
  virtual ~Axis(){}
private:
  Axis(const Axis& a_from):istorable(a_from),m_axis(a_from.m_axis){}
  Axis& operator=(const Axis&){return *this;}
protected:
  const histo::axis<double,unsigned int>& m_axis;
};


inline bool Axis_read(iobj_visitor& a_visitor,histo::axis<double,unsigned int>& a_axis){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(!a_visitor.visit(a_axis.m_offset)) return false;

 {int nbin;
  if(!a_visitor.visit(nbin)) return false;
  a_axis.m_number_of_bins = nbin;}

  if(!a_visitor.visit(a_axis.m_minimum_value)) return false;
  if(!a_visitor.visit(a_axis.m_maximum_value)) return false;
  if(!a_visitor.visit(a_axis.m_fixed)) return false;
  if(!a_visitor.visit(a_axis.m_bin_width)) return false;
  if(!a_visitor.visit(a_axis.m_edges)) return false;

  //if(!a_visitor.end(*this)) return false;
  return true;
}

class Item : public virtual istorable {
public:
  TOOLS_SCLASS(tools::osc::Item)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<Item>(this,a_class)) return p;
    return 0;
  }
public:
  virtual const std::string& store_cls() const {return s_item();}
  virtual bool visit(iobj_const_visitor& a_visitor) const {
    if(!a_visitor.begin(*this,s_item(),Item::s_visit)) return false;

    int version = 1;
    if(!a_visitor.visit("fVersion",version)) return false;

    if(!a_visitor.visit("fKey",fKey)) return false;
    if(!a_visitor.visit("fValue",fValue)) return false;
    if(!a_visitor.visit("fSticky",fSticky)) return false;

    if(!a_visitor.end(*this)) return false;
    return true;
  }
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const Item* local = safe_cast<istorable,Item>(a_o);
    if(!local) return false;
    return local->Item::visit(a_v); //IMPORTANT : have Item::
  }
public:
  virtual bool read(iobj_visitor& a_visitor) {
    //if(!a_visitor.begin(*this)) return false;

    int version;
    if(!a_visitor.visit(version)) return false;

    if(!a_visitor.visit(fKey)) return false;
    if(!a_visitor.visit(fValue)) return false;
    if(!a_visitor.visit(fSticky)) return false;

    //if(!a_visitor.end(*this)) return false;
    return true;
  }
public:
  Item(){}
  Item(const std::string& aKey,const std::string& aValue,bool aSticky)
  :fKey(aKey),fValue(aValue),fSticky(aSticky){}
  virtual ~Item(){}
public:
  Item(const Item& a_from)
  :istorable(a_from)
  ,fKey(a_from.fKey)
  ,fValue(a_from.fValue)
  ,fSticky(a_from.fSticky)
  {}
  Item& operator=(const Item& a_from) {
    if(&a_from==this) return *this;
    fKey = a_from.fKey;
    fValue = a_from.fValue;
    fSticky = a_from.fSticky;
    return *this;
  }
public:
  std::string fKey;
  std::string fValue;
  bool fSticky;
};


template <class T>
class Vector : public virtual istorable {
public:
  TOOLS_T_SCLASS(T,tools::osc::Vector)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<Vector>(this,a_class)) return p;
    return 0;
  }
  virtual const std::string& store_cls() const {
    static const std::string s_v("BatchLab::Vector<"+m_T+">");
    return s_v;
  }
  virtual bool visit(iobj_const_visitor& a_v) const {
    if(!a_v.begin(*this,Vector<T>::store_cls(),Vector<T>::s_visit)) return false;

    int version = 1;
    if(!a_v.visit("fVersion",version)) return false;

    unsigned int number = (unsigned int)m_vec.size();
    if(!a_v.visit("fSize",number)) return false;

    std::string stmp;
    for(unsigned int index=0;index<number;index++) {
      const T& elem = m_vec[index];
      if(!num2s(index,stmp)) return false;
      if(!a_v.visit(stmp,elem)) return false;
    }

    if(!a_v.end(*this)) return false;
    return true;
  }
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const Vector* local = safe_cast<istorable,Vector>(a_o);
    if(!local) return false;
    return local->Vector<T>::visit(a_v); //IMPORTANT : have Vector::
  }
public:
  Vector(const std::vector<T>& a_vec,const std::string& a_T)
  :m_vec(a_vec),m_T(a_T){}
  virtual ~Vector(){}
private:
  Vector(const Vector& a_from)
  :istorable(a_from),m_vec(a_from.m_vec),m_T(a_from.m_T){}
  Vector& operator=(const Vector&){return *this;}
protected:
  const std::vector<T>& m_vec;
  std::string m_T;
};


template <class T>
inline bool std_vector_read(iobj_visitor& a_visitor,std::vector<T>& a_vec) {
  a_vec.clear();

  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  unsigned int number;
  if(!a_visitor.visit(number)) return false;

  a_vec.resize(number);
  for(unsigned int index=0;index<number;index++) {
    T& elem = a_vec[index];
    if(!elem.read(a_visitor)) return false;
  }

  //if(!a_visitor.end(*this)) return false;
  return true;
}

class Annotation : public virtual istorable {
public:
  TOOLS_SCLASS(tools::osc::Annotation)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<Annotation>(this,a_class)) return p;
    return 0;
  }
  virtual const std::string& store_cls() const {return s_annotation();}
  virtual bool visit(iobj_const_visitor& a_v) const {
    if(!a_v.begin(*this,s_annotation(),Annotation::s_visit)) return false;

    int version = 1;
    if(!a_v.visit("fVersion",version)) return false;

    Vector<Item> v(m_items,s_item());
    if(!a_v.visit("fItems",v)) return false;

    if(!a_v.end(*this)) return false;
    return true;
  }
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const Annotation* local = safe_cast<istorable,Annotation>(a_o);
    if(!local) return false;
    return local->Annotation::visit(a_v); //IMPORTANT : have Annotation::
  }
public:
  Annotation(){}
  virtual ~Annotation(){}
private:
  Annotation(const Annotation& a_from):istorable(a_from){}
  Annotation& operator=(const Annotation&){return *this;}
public:
  std::vector<Item> m_items;
};

inline bool Annotation_read(iobj_visitor& a_visitor) {
  //if(!a_visitor.begin(*this)) return false;
  int version;
  if(!a_visitor.visit(version)) return false;
  std::vector<Item> fItems;
  if(!std_vector_read<Item>(a_visitor,fItems)) return false;
  //if(!a_visitor.end(*this)) return false;
  return true;
}

inline void map2vec(const std::map<std::string,std::string>& a_in,std::vector<Item>& a_out) {
  a_out.clear();
  std::map<std::string,std::string>::const_iterator it;
  for(it=a_in.begin();it!=a_in.end();++it) {
    a_out.push_back(Item((*it).first,(*it).second,false));
  }
}

typedef histo::histo_data<double,unsigned int,unsigned int,double> hd_data;

template <class HIST>
class BaseHistogram : public virtual istorable {
public:
  TOOLS_T_SCLASS(HIST,tools::osc::BaseHistogram)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<BaseHistogram>(this,a_class)) return p;
    return 0;
  }
public:
  virtual const std::string& store_cls() const {return s_base_histogram();}
  virtual bool visit(iobj_const_visitor& a_v) const {
    if(!a_v.begin(*this,s_base_histogram(),BaseHistogram::s_visit)) return false;

    int version = 1;
    if(!a_v.visit("fVersion",version)) return false;

    Annotation ano;
    map2vec(m_hist.annotations(),ano.m_items);
    if(!a_v.visit("fAnnotation",ano)) return false;

    if(!a_v.end(*this)) return false;
    return true;
  }
protected:
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const BaseHistogram* local =
      safe_cast<istorable,BaseHistogram>(a_o);
    if(!local) return false;
    return local->BaseHistogram::visit(a_v); //IMPORTANT : have BaseHistogram::
  }
public:
  BaseHistogram(const HIST& a_hist):m_hist(a_hist){}
  virtual ~BaseHistogram(){}
private:
  BaseHistogram(const BaseHistogram& a_from)
  :istorable(a_from),m_hist(a_from.m_hist)
  {}
  BaseHistogram& operator=(const BaseHistogram&){return *this;}
protected:
  const HIST& m_hist;
};

inline bool BaseHistogram_read(iobj_visitor& a_visitor){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(!Annotation_read(a_visitor)) return false;

  //if(!a_visitor.end(*this)) return false;
  return true;
}

template <class HIST>
inline bool visitHistogram(const HIST& a_hist,iobj_const_visitor& a_visitor){
  if(!a_visitor.visit("fTitle",a_hist.title())) return false;
  if(!a_visitor.visit("fDimension",(int)a_hist.dimension())) return false;
  if(!a_visitor.visit("fBinNumber",(int)a_hist.get_bins())) return false;

  std::vector<int> bins_entries;
  convert<unsigned int,int>(a_hist.bins_entries(),bins_entries);
  if(!a_visitor.visit("fBinEntries",bins_entries)) return false;

  if(!a_visitor.visit("fBinSw",a_hist.bins_sum_w())) return false;
  if(!a_visitor.visit("fBinSw2",a_hist.bins_sum_w2())) return false;
  if(!a_visitor.visit("fBinSxw",a_hist.bins_sum_xw())) return false;
  if(!a_visitor.visit("fBinSx2w",a_hist.bins_sum_x2w())) return false;
 {std::string name;
  for(unsigned int iaxis=0;iaxis<a_hist.dimension();iaxis++) {
    if(!num2s(iaxis,name)) return false;
    name = "fAxes_"+name;
    Axis axis(a_hist.get_axis(iaxis));
    if(!a_visitor.visit(name,axis)) return false;
  }}
 {int dummy = 0;
  if(!a_visitor.visit("fMode",dummy)) return false;} //m_mode
  if(!a_visitor.visit("fProfile",false)) return false;
 {std::vector<double> dummy;
  if(!a_visitor.visit("fBinSvw",dummy)) return false;
  if(!a_visitor.visit("fBinSv2w",dummy)) return false;}
  if(!a_visitor.visit("fCutV",false)) return false;
 {double dummy = 0;
  if(!a_visitor.visit("fMinV",dummy)) return false;
  if(!a_visitor.visit("fMaxV",dummy)) return false;}
  // Not written :
  //a_hist.fDoubles
  //a_hist.fInts
  return true;
}

inline bool readHistogram(hd_data& aData,iobj_visitor& a_visitor){
  if(!a_visitor.visit(aData.m_title)) return false;
 {int dim;
  if(!a_visitor.visit(dim)) return false;
  aData.m_dimension = dim;}

 {int nbin;
  if(!a_visitor.visit(nbin)) return false;
  aData.m_bin_number = nbin;}

 {std::vector<int> vec;
  if(!a_visitor.visit(vec)) return false;
  convert<int,unsigned int>(vec,aData.m_bin_entries);}

  if(!a_visitor.visit(aData.m_bin_Sw)) return false;
  if(!a_visitor.visit(aData.m_bin_Sw2)) return false;
  if(!a_visitor.visit(aData.m_bin_Sxw)) return false;
  if(!a_visitor.visit(aData.m_bin_Sx2w)) return false;
  aData.m_axes.clear();
  for(unsigned int iaxis=0;iaxis<aData.m_dimension;iaxis++) {
    histo::axis<double,unsigned int> baxis;
    if(!Axis_read(a_visitor,baxis)) return false;
    aData.m_axes.push_back(baxis);
  }
 {int dummy;
  if(!a_visitor.visit(dummy)) return false;} //m_mode

 {bool dummy;
  if(!a_visitor.visit(dummy)) return false;} //m_is_profile

 {std::vector<double> dummy;
  if(!a_visitor.visit(dummy)) return false;} //m_bin_Svw

 {std::vector<double> dummy;
  if(!a_visitor.visit(dummy)) return false;} //m_bin_Sv2w

 {bool dummy;
  if(!a_visitor.visit(dummy)) return false;} //m_cut_v

 {double dummy;
  if(!a_visitor.visit(dummy)) return false;} //aData.m_min_v

 {double dummy;
  if(!a_visitor.visit(dummy)) return false;} //aData.m_max_v

  //aData.fDoubles
  //aData.fInts
  //aData.m_coords.resize(aData.m_dimension,0);
  //aData.m_ints.resize(aData.m_dimension,0);

  return true;
}

template <class HIST>
class Histogram : public virtual istorable {
public:
  TOOLS_T_SCLASS(HIST,tools::osc::Histogram)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<Histogram>(this,a_class)) return p;
    return 0;
  }
public:
  virtual const std::string& store_cls() const {return m_cls;}
  virtual bool visit(iobj_const_visitor& a_v) const {
    if(!a_v.begin(*this,m_cls,Histogram::s_visit)) return false;

    int version = 1;
    if(!a_v.visit("fVersion",version)) return false;

    BaseHistogram<HIST> bh(m_hist);
    if(!bh.visit(a_v)) return false;

    if(!visitHistogram(m_hist,a_v)) return false;

    if(!a_v.end(*this)) return false;
    return true;
  }
protected:
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const Histogram* local = safe_cast<istorable,Histogram>(a_o);
    if(!local) return false;
    return local->Histogram::visit(a_v); //IMPORTANT : have Histogram::
  }
public:
  Histogram(const HIST& a_hist,const std::string& a_cls)
  :m_hist(a_hist),m_cls(a_cls){}
  virtual ~Histogram(){}
public:
  Histogram(const Histogram& a_from)
  :istorable(a_from)
  ,m_hist(a_from.m_hist)
  ,m_cls(a_from.m_cls)
  {}
  Histogram& operator=(const Histogram& a_from){
    m_cls = a_from.m_cls;
    return *this;
  }
protected:
  const HIST& m_hist;
  std::string m_cls;
};

inline bool visit(iobj_const_visitor& a_v,const histo::h1d& a_histo) {
  Histogram<histo::h1d> h(a_histo,s_h1d());
  return h.visit(a_v);
}

inline bool read(iobj_visitor& a_visitor,histo::h1d& a_histo){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(version!=1) {
    //this may come from an unexpected byteswap.
    a_visitor.out() << "tools::osc::read :"
                    << " unexpected version " << version
                    << std::endl;
    return false;
  }

  if(!BaseHistogram_read(a_visitor)) return false;

  hd_data hdata;
  if(!readHistogram(hdata,a_visitor)) return false;

  hdata.update_fast_getters();
  a_histo.copy_from_data(hdata);

  //fAxis.copy(fHistogram.get_axis(0));

  //if(!a_visitor.end(*this)) return false;

  return true;
}

inline bool visit(iobj_const_visitor& a_v,const histo::h2d& a_histo) {
  Histogram<histo::h2d> h(a_histo,s_h2d());
  return h.visit(a_v);
}

inline bool read(iobj_visitor& a_visitor,histo::h2d& a_histo){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(version!=1) {
    //this may come from an unexpected byteswap.
    a_visitor.out() << "tools::osc::read :"
                    << " unexpected version " << version
                    << std::endl;
    return false;
  }

  if(!BaseHistogram_read(a_visitor)) return false;

  hd_data hdata;
  if(!readHistogram(hdata,a_visitor)) return false;

  hdata.update_fast_getters();
  a_histo.copy_from_data(hdata);

  //fAxisX.copy(fHistogram.get_axis(0));
  //fAxisY.copy(fHistogram.get_axis(1));

  //if(!a_visitor.end(*this)) return false;

  return true;
}

inline bool visit(iobj_const_visitor& a_v,const histo::h3d& a_histo) {
  Histogram<histo::h3d> h(a_histo,s_h3d());
  return h.visit(a_v);
}

inline bool read(iobj_visitor& a_visitor,histo::h3d& a_histo){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(version!=1) {
    //this may come from an unexpected byteswap.
    a_visitor.out() << "tools::osc::read :"
                    << " unexpected version " << version
                    << std::endl;
    return false;
  }

  if(!BaseHistogram_read(a_visitor)) return false;

  hd_data hdata;
  if(!readHistogram(hdata,a_visitor)) return false;
  hdata.update_fast_getters();
  a_histo.copy_from_data(hdata);

  //fAxisX.copy(fHistogram.get_axis(0));
  //fAxisY.copy(fHistogram.get_axis(1));
  //fAxisZ.copy(fHistogram.get_axis(2));

  //if(!a_visitor.end(*this)) return false;

  return true;
}

typedef histo::profile_data<double,unsigned int,unsigned int,double,double> pd_data;

template <class PROF>
inline bool visitProfile(const PROF& a_prof,iobj_const_visitor& a_visitor){
  if(!a_visitor.visit("fTitle",a_prof.title())) return false;
  if(!a_visitor.visit("fDimension",(int)a_prof.dimension())) return false;
  if(!a_visitor.visit("fBinNumber",(int)a_prof.get_bins())) return false;

  std::vector<int> bins_entries;
  convert<unsigned int,int>(a_prof.bins_entries(),bins_entries);
  if(!a_visitor.visit("fBinEntries",bins_entries)) return false;

  if(!a_visitor.visit("fBinSw",a_prof.bins_sum_w())) return false;
  if(!a_visitor.visit("fBinSw2",a_prof.bins_sum_w2())) return false;
  if(!a_visitor.visit("fBinSxw",a_prof.bins_sum_xw())) return false;
  if(!a_visitor.visit("fBinSx2w",a_prof.bins_sum_x2w())) return false;
  std::string name;
  for(unsigned int iaxis=0;iaxis<a_prof.dimension();iaxis++) {
    if(!num2s(iaxis,name)) return false;
    name = "fAxes_"+name;
    Axis axis(a_prof.get_axis(iaxis));
    if(!a_visitor.visit(name,axis)) return false;
  }
 {int dummy = 0;
  if(!a_visitor.visit("fMode",dummy)) return false;} //m_mode
  if(!a_visitor.visit("fProfile",true)) return false;
  if(!a_visitor.visit("fBinSvw",a_prof.bins_sum_vw())) return false;
  if(!a_visitor.visit("fBinSv2w",a_prof.bins_sum_v2w())) return false;
  if(!a_visitor.visit("fCutV",a_prof.cut_v())) return false;
  if(!a_visitor.visit("fMinV",a_prof.min_v())) return false;
  if(!a_visitor.visit("fMaxV",a_prof.max_v())) return false;
  // Not written :
  //a_prof.fDoubles
  //a_prof.fInts
  return true;
}

template <class PROF>
class Profile : public virtual istorable {
public:
  TOOLS_T_SCLASS(PROF,tools::osc::Profile)
protected:
  virtual void* cast(const std::string& a_class) const {
    if(void* p = cmp_cast<Profile>(this,a_class)) return p;
    return 0;
  }
public:
  virtual const std::string& store_cls() const {return m_cls;}
  virtual bool visit(iobj_const_visitor& a_v) const {
    if(!a_v.begin(*this,m_cls,Profile::s_visit)) return false;

    int version = 1;
    if(!a_v.visit("fVersion",version)) return false;

    BaseHistogram<PROF> bh(m_hist);
    if(!bh.visit(a_v)) return false;

    if(!visitProfile(m_hist,a_v)) return false;

    if(!a_v.end(*this)) return false;
    return true;
  }
protected:
  static bool s_visit(const istorable& a_o,iobj_const_visitor& a_v){
    const Profile* local = safe_cast<istorable,Profile>(a_o);
    if(!local) return false;
    return local->Profile::visit(a_v); //IMPORTANT : have Profile::
  }
public:
  Profile(const PROF& a_hist,const std::string& a_cls)
  :m_hist(a_hist),m_cls(a_cls){}
  virtual ~Profile(){}
public:
  Profile(const Profile& a_from)
  :istorable(a_from)
  ,m_hist(a_from.m_hist)
  ,m_cls(a_from.m_cls)
  {}
  Profile& operator=(const Profile& a_from){
    m_cls = a_from.m_cls;
    return *this;
  }
protected:
  const PROF& m_hist;
  std::string m_cls;
};

inline bool readProfile(pd_data& aData,iobj_visitor& a_visitor){
  if(!a_visitor.visit(aData.m_title)) return false;

 {int dim;
  if(!a_visitor.visit(dim)) return false;
  aData.m_dimension = dim;}

 {int nbin;
  if(!a_visitor.visit(nbin)) return false;
  aData.m_bin_number = nbin;}

 {std::vector<int> vec;
  if(!a_visitor.visit(vec)) return false;
  convert<int,unsigned int>(vec,aData.m_bin_entries);}

  if(!a_visitor.visit(aData.m_bin_Sw)) return false;
  if(!a_visitor.visit(aData.m_bin_Sw2)) return false;
  if(!a_visitor.visit(aData.m_bin_Sxw)) return false;
  if(!a_visitor.visit(aData.m_bin_Sx2w)) return false;
  aData.m_axes.clear();
  for(unsigned int iaxis=0;iaxis<aData.m_dimension;iaxis++) {
    histo::axis<double,unsigned int> baxis;
    if(!Axis_read(a_visitor,baxis)) return false;
    aData.m_axes.push_back(baxis);
  }
 {int dummy;
  if(!a_visitor.visit(dummy)) return false;} //m_mode
  if(!a_visitor.visit(aData.m_is_profile)) return false;
  if(!a_visitor.visit(aData.m_bin_Svw)) return false;
  if(!a_visitor.visit(aData.m_bin_Sv2w)) return false;
  if(!a_visitor.visit(aData.m_cut_v)) return false;
  if(!a_visitor.visit(aData.m_min_v)) return false;
  if(!a_visitor.visit(aData.m_max_v)) return false;

  // Not written :
  //aData.fDoubles
  //aData.fInts
  //aData.m_coords.resize(aData.m_dimension,0);
  //aData.m_ints.resize(aData.m_dimension,0);

  return true;
}

inline bool visit(iobj_const_visitor& a_v,const histo::p1d& a_histo) {
  Profile<histo::p1d> h(a_histo,s_p1d());
  return h.visit(a_v);
}

inline bool read(iobj_visitor& a_visitor,histo::p1d& a_histo){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(version!=1) {
    //this may come from an unexpected byteswap.
    a_visitor.out() << "tools::osc::read :"
                    << " unexpected version " << version
                    << std::endl;
    return false;
  }


  if(!BaseHistogram_read(a_visitor)) return false;

  pd_data hdata;
  if(!readProfile(hdata,a_visitor)) return false;
  hdata.update_fast_getters();
  a_histo.copy_from_data(hdata);

  //fAxis.copy(fHistogram.get_axis(0));

  //if(!a_visitor.end(*this)) return false;

  return true;
}

inline bool visit(iobj_const_visitor& a_v,const histo::p2d& a_histo) {
  Profile<histo::p2d> h(a_histo,s_p2d());
  return h.visit(a_v);
}

inline bool read(iobj_visitor& a_visitor,histo::p2d& a_histo){
  //if(!a_visitor.begin(*this)) return false;

  int version;
  if(!a_visitor.visit(version)) return false;

  if(version!=1) {
    //this may come from an unexpected byteswap.
    a_visitor.out() << "tools::osc::read :"
                    << " unexpected version " << version
                    << std::endl;
    return false;
  }


  if(!BaseHistogram_read(a_visitor)) return false;

  pd_data hdata;
  if(!readProfile(hdata,a_visitor)) return false;
  hdata.update_fast_getters();
  a_histo.copy_from_data(hdata);

  //fAxisX.copy(a_histo.get_axis(0));
  //fAxisY.copy(a_histo.get_axis(1));

  //if(!a_visitor.end(*this)) return false;

  return true;
}

template <class HIST>
class Histogram_cp : public Histogram<HIST> {
  typedef Histogram<HIST> parent;
public:
  Histogram_cp(const HIST& a_h,const std::string& a_cls)
  :parent(m_cp,a_cls) //give ref of m_cp to Histogram.
  ,m_cp(a_h)   //do a local copy.
  //WARNING : the upper is ok as long as Histogram constructor does nothing
  //          else than keeping the ref to m_cp. Else it would do
  //          something on an empty histo (and not on a copy of the
  //          passed a_data).
  {}
  virtual ~Histogram_cp(){}
public:
  Histogram_cp(const Histogram_cp& a_from)
  :istorable(a_from)
  ,parent(m_cp,a_from.m_cls)
  ,m_cp(a_from.m_cp)
  {}
  Histogram_cp& operator=(const Histogram_cp& a_from){
    parent::operator=(a_from);
    m_cp = a_from.m_cp;
    return *this;
  }
protected:
  HIST m_cp;
};

template <class PROF>
class Profile_cp : public Profile<PROF> {
  typedef Profile<PROF> parent;
public:
  Profile_cp(const PROF& a_p,const std::string& a_cls)
  :parent(m_cp,a_cls) //give ref of m_cp to Profile.
  ,m_cp(a_p)   //do a local copy.
  //WARNING : the upper is ok as long as Profile constructor does nothing
  //          else than keeping the ref to m_cp. Else it would do
  //          something on an empty histo (and not on a copy of the
  //          passed a_data).
  {}
  virtual ~Profile_cp(){}
public:
  Profile_cp(const Profile_cp& a_from)
  :istorable(a_from)
  ,parent(m_cp,a_from.m_cls)
  ,m_cp(a_from.m_cp)
  {}
  Profile_cp& operator=(const Profile_cp& a_from){
    parent::operator=(a_from);
    m_cp = a_from.m_cp;
    return *this;
  }
protected:
  PROF m_cp;
};

}}

#endif
