xlistnode.h
1 /***************************************************************************
2  Copyright (C) 2002-2015 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13 ***************************************************************************/
14 #ifndef XLISTNODE_H_
15 #define XLISTNODE_H_
16 
17 #include "xnode.h"
18 #include <functional>
19 
20 class DECLSPEC_KAME XListNodeBase : public XNode {
21 public:
22  explicit XListNodeBase(const char *name, bool runtime = false);
23  virtual ~XListNodeBase() = default;
24 
25  //! Create a object, whose class is determined from \a type.
26  //! Scripting only. Use XNode::create for coding instead.
27  virtual shared_ptr<XNode> createByTypename(
28  const XString &type, const XString &name) = 0;
29 
30  virtual bool isThreadSafeDuringCreationByTypename() const = 0;
31 
32  struct DECLSPEC_KAME Payload : public XNode::Payload {
33  Talker<XListNodeBase*> &onListChanged() {return m_tlkOnListChanged;}
34  struct MoveEvent {
35  unsigned int src_idx, dst_idx;
36  XListNodeBase *emitter;
37  };
38  Talker<MoveEvent> &onMove() {return m_tlkOnMove;}
39  const Talker<MoveEvent> &onMove() const {return m_tlkOnMove;}
40  struct CatchEvent {
41  XListNodeBase *emitter;
42  shared_ptr<XNode> caught;
43  int index;
44  };
45  Talker<CatchEvent> &onCatch() {return m_tlkOnCatch;}
46  const Talker<CatchEvent> &onCatch() const {return m_tlkOnCatch;}
47  struct ReleaseEvent {
48  XListNodeBase *emitter;
49  shared_ptr<XNode> released;
50  int index;
51  };
52  Talker<ReleaseEvent> &onRelease() {return m_tlkOnRelease;}
53  const Talker<ReleaseEvent> &onRelease() const {return m_tlkOnRelease;}
54  private:
55  TalkerSingleton<XListNodeBase*> m_tlkOnListChanged;
56  Talker<MoveEvent> m_tlkOnMove;
57  Talker<CatchEvent> m_tlkOnCatch;
58  Talker<ReleaseEvent> m_tlkOnRelease;
59  virtual void catchEvent(const shared_ptr<XNode>&, int);
60  virtual void releaseEvent(const shared_ptr<XNode>&, int);
61  virtual void moveEvent(unsigned int src_idx, unsigned int dst_idx);
62  virtual void listChangeEvent();
63  };
64 protected:
65 };
66 
67 //! List node for simples nodes, such like XIntNode.
68 template <class NT>
69 class XListNode : public XListNodeBase {
70 public:
71  explicit XListNode(const char *name, bool runtime = false)
72  : XListNodeBase(name, runtime) {}
73  virtual ~XListNode() = default;
74 
75  virtual bool isThreadSafeDuringCreationByTypename() const {return true;}
76 
77  virtual shared_ptr<XNode> createByTypename(
78  const XString &, const XString &name) {
79  return this->create<NT>(name.c_str(), false);
80  }
81 };
82 
83 //! creation by UI is not allowed.
84 template <class NT>
85 class XAliasListNode : public XListNodeBase {
86 public:
87  explicit XAliasListNode(const char *name, bool runtime = false)
88  : XListNodeBase(name, runtime) {}
89  virtual ~XAliasListNode() = default;
90 
91  virtual bool isThreadSafeDuringCreationByTypename() const {return true;}
92 
93  virtual shared_ptr<XNode> createByTypename(
94  const XString &, const XString &) {
95  return shared_ptr<XNode>();
96  }
97 };
98 
99 template <class NT>
101 public:
102  explicit XCustomTypeListNode(const char *name, bool runtime = false)
103  : XListNodeBase(name, runtime) {}
104  virtual ~XCustomTypeListNode() = default;
105 
106  virtual bool isThreadSafeDuringCreationByTypename() const {return false;} //! default behavior for safety.
107 
108  virtual shared_ptr<XNode> createByTypename(
109  const XString &type, const XString &name) = 0;
110 };
111 
112 #include <functional>
113 
114 //! Register typename and constructor.
115 //! make static member of TypeHolder<> in your class
116 //! After def. of static TypeHolder<>, define Creator to register.
117 //! call creator(type)(type, name, ...) to create children.
118 template <class... ArgTypes>
119 struct XTypeHolder {
120  using creator_t = std::function<shared_ptr<XNode>(const char*, bool, ArgTypes&&...)>;
121 
122  XTypeHolder() {
123  fprintf(stderr, "New typeholder\n");
124  }
125 
126  creator_t creator(const XString &tp) {
127  for(unsigned int i = 0; i < names.size(); i++) {
128  if(names[i] == tp) return creators[i];
129  }
130  return [](const char*, bool, ArgTypes&&...){return shared_ptr<XNode>();}; //empty
131  }
132  template <class tChild>
133  struct Creator {
134  Creator(XTypeHolder &holder, const char *name, const char *label = 0L) {
135  creator_t create_typed =
136  [](const char *name, bool runtime, ArgTypes&&... args)->shared_ptr<XNode>
137  {return XNode::createOrphan<tChild>(name, runtime, std::forward<ArgTypes>(args)...);};
138  assert(create_typed);
139  if( !label)
140  label = name;
141  if(std::find(holder.names.begin(), holder.names.end(), XString(name)) != holder.names.end()) {
142  fprintf(stderr, "Duplicated name!\n");
143  return;
144  }
145  holder.creators.push_back(create_typed);
146  holder.names.push_back(XString(name));
147  holder.labels.push_back(XString(label));
148  fprintf(stderr, "%s %s\n", name, label);
149  }
150  private:
151  template <class T>
152  static shared_ptr<XNode> creator_(const char *name, bool runtime, ArgTypes&&... args) {
153  return XNode::createOrphan<T>(name, runtime, std::forward<ArgTypes>(args)...);
154  }
155  };
156  template <class tChild> friend struct Creator;
157  std::deque<creator_t> creators;
158  std::deque<XString> names, labels;
159 };
160 
161 #define DEFINE_TYPE_HOLDER(...) \
162  typedef XTypeHolder<__VA_ARGS__> TypeHolder; \
163  static TypeHolder s_types; \
164  static TypeHolder::creator_t creator(const XString &tp) {return s_types.creator(tp);} \
165  static std::deque<XString> &typenames() {return s_types.names;} \
166  static std::deque<XString> &typelabels() {return s_types.labels;}
167 
168 #define DECLARE_TYPE_HOLDER(list) \
169  list::TypeHolder list::s_types;
170 
171 #define REGISTER_TYPE_2__(list, type, name, label) list::TypeHolder::Creator<type> \
172  g_driver_type_ ## name(list::s_types, # name, label);
173 
174 #define REGISTER_TYPE(list, type, label) REGISTER_TYPE_2__(list, X ## type, type, label)
175 
176 class DECLSPEC_KAME XStringList : public XListNode<XStringNode> {
177 public:
178  explicit XStringList(const char *name, bool runtime = false)
179  : XListNode<XStringNode>(name, runtime) {}
180  virtual ~XStringList() = default;
181 };
182 
183 #endif /*XLISTNODE_H_*/

Generated for KAME4 by  doxygen 1.8.3