#ifndef _FF_FACTORY_H_
#define _FF_FACTORY_H_

// FsaF* vs F* (regular ff/factory).

//TODO: use http://www.boost.org/doc/libs/1_43_0/libs/functional/factory/doc/html/index.html ?

/*TODO: register state identity separately from feature function identity?  as
 * in: string registry for name of state somewhere, assert that same result is
 * computed by all users?  or, we can just require that ff sharing same state
 * all be mashed into a single ffunc, which can just emit all the fid scores at
 * once.  that's fine.
 */


#include <iostream>
#include <string>
#include <map>
#include <stdexcept>

#include <boost/shared_ptr.hpp>

#include "ff_fsa_dynamic.h"

class FeatureFunction;

class FsaFeatureFunction;


struct UntypedFactory {
  virtual ~UntypedFactory();
  virtual std::string usage(bool params,bool verbose) const = 0;
};

template <class FF>
struct FactoryBase : public UntypedFactory {
  typedef FF F;
  typedef boost::shared_ptr<F> FP;

  virtual FP Create(std::string param) const = 0;
};

/* see cdec_ff.cc for example usage: this create concrete factories to be registered */
template<class FF>
struct FFFactory : public FactoryBase<FeatureFunction> {
  FP Create(std::string param) const {
    FF *ret=new FF(param);
    ret->Init();
    return FP(ret);
  }
  virtual std::string usage(bool params,bool verbose) const {
    return FF::usage(params,verbose);
  }
};


// same as above, but we didn't want to require a typedef e.g. Parent in FF class, and template typedef isn't available
template<class FF>
struct FsaFactory : public FactoryBase<FsaFeatureFunction> {
  FP Create(std::string param) const {
    FF *ret=new FF(param);
    ret->Init();
    return FP(ret);
  }
  virtual std::string usage(bool params,bool verbose) const {
    return FF::usage(params,verbose);
  }
};

struct UntypedFactoryRegistry {
  std::string usage(std::string const& ffname,bool params=true,bool verbose=true) const;
  bool have(std::string const& ffname);
  void DisplayList() const;
  void Register(const std::string& ffname, UntypedFactory* factory);
  void Register(UntypedFactory* factory);
  void clear();
  static bool parse_debug(std::string & param_in_out); // returns true iff param starts w/ debug (and remove that prefix from param)
 protected:
  typedef boost::shared_ptr<UntypedFactory> FactoryP;
  typedef std::map<std::string, FactoryP > Factmap;
  Factmap reg_;
  friend int main(int argc, char** argv);
  friend class UntypedFactory;
};



template <class Feat>
struct FactoryRegistry : public UntypedFactoryRegistry {
  typedef Feat F;
  typedef boost::shared_ptr<F> FP;
  typedef FactoryBase<F> FB;

  FP Create(const std::string& ffname, std::string param) const {
    using namespace std;
    Factmap::const_iterator it = reg_.find(ffname);
    if (it == reg_.end())
      throw std::runtime_error("I don't know how to create feature "+ffname);
    bool debug=parse_debug(param);
    if (debug)
      cerr<<"debug enabled for "<<ffname<< " - remaining options: '"<<param<<"'\n";
    FP res = dynamic_cast<FB const&>(*it->second).Create(param);
    res->init_name_debug(ffname,debug);
    // could add a res->Init() here instead of in Create if we wanted feature id to potentially differ based on the registered name rather than static usage() - of course, specific feature ids can be computed on the basis of feature param as well; this only affects the default single feature id=name
    return res;
  }
};

typedef FactoryRegistry<FeatureFunction> FFRegistry;
typedef FactoryRegistry<FsaFeatureFunction> FsaFFRegistry;

extern FsaFFRegistry fsa_ff_registry;
inline FsaFFRegistry & global_fsa_ff_registry() { return fsa_ff_registry; }
extern FFRegistry ff_registry;
inline FFRegistry & global_ff_registry() { return ff_registry; }

void ff_usage(std::string const& name,std::ostream &out=std::cout);

/*
extern boost::shared_ptr<FsaFFRegistry> global_fsa_ff_registry;
extern boost::shared_ptr<FFRegistry> global_ff_registry;
*/
#endif