ctkAbstractFactory.tpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) Kitware Inc.
  4. Licensed under the Apache License, Version 2.0 (the "License");
  5. you may not use this file except in compliance with the License.
  6. You may obtain a copy of the License at
  7. http://www.apache.org/licenses/LICENSE-2.0.txt
  8. Unless required by applicable law or agreed to in writing, software
  9. distributed under the License is distributed on an "AS IS" BASIS,
  10. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. See the License for the specific language governing permissions and
  12. limitations under the License.
  13. =========================================================================*/
  14. #ifndef __ctkAbstractFactory_tpp
  15. #define __ctkAbstractFactory_tpp
  16. // QT includes
  17. #include <QDebug>
  18. // CTK includes
  19. #include "ctkAbstractFactory.h"
  20. //----------------------------------------------------------------------------
  21. // ctkAbstractFactoryItem methods
  22. //----------------------------------------------------------------------------
  23. template<typename BaseClassType>
  24. ctkAbstractFactoryItem<BaseClassType>::ctkAbstractFactoryItem()
  25. :Instance()
  26. {
  27. this->Verbose = false;
  28. }
  29. //----------------------------------------------------------------------------
  30. template<typename BaseClassType>
  31. QString ctkAbstractFactoryItem<BaseClassType>::loadErrorString()const
  32. {
  33. return QString();
  34. }
  35. //----------------------------------------------------------------------------
  36. template<typename BaseClassType>
  37. QStringList ctkAbstractFactoryItem<BaseClassType>::instantiateErrorStrings()const
  38. {
  39. return this->InstantiateErrorStrings;
  40. }
  41. //----------------------------------------------------------------------------
  42. template<typename BaseClassType>
  43. void ctkAbstractFactoryItem<BaseClassType>::appendInstantiateErrorString(const QString& errorString)
  44. {
  45. this->InstantiateErrorStrings << errorString;
  46. }
  47. //----------------------------------------------------------------------------
  48. template<typename BaseClassType>
  49. void ctkAbstractFactoryItem<BaseClassType>::clearInstantiateErrorStrings()
  50. {
  51. this->InstantiateErrorStrings.clear();
  52. }
  53. //----------------------------------------------------------------------------
  54. template<typename BaseClassType>
  55. QStringList ctkAbstractFactoryItem<BaseClassType>::instantiateWarningStrings()const
  56. {
  57. return this->InstantiateWarningStrings;
  58. }
  59. //----------------------------------------------------------------------------
  60. template<typename BaseClassType>
  61. void ctkAbstractFactoryItem<BaseClassType>::appendInstantiateWarningString(const QString& msg)
  62. {
  63. this->InstantiateWarningStrings << msg;
  64. }
  65. //----------------------------------------------------------------------------
  66. template<typename BaseClassType>
  67. void ctkAbstractFactoryItem<BaseClassType>::clearInstantiateWarningStrings()
  68. {
  69. this->InstantiateWarningStrings.clear();
  70. }
  71. //----------------------------------------------------------------------------
  72. template<typename BaseClassType>
  73. BaseClassType* ctkAbstractFactoryItem<BaseClassType>::instantiate()
  74. {
  75. this->clearInstantiateErrorStrings();
  76. this->clearInstantiateWarningStrings();
  77. if (this->Instance)
  78. {
  79. return this->Instance;
  80. }
  81. this->Instance = this->instanciator();
  82. return this->Instance;
  83. }
  84. //----------------------------------------------------------------------------
  85. template<typename BaseClassType>
  86. bool ctkAbstractFactoryItem<BaseClassType>::instantiated()const
  87. {
  88. return (this->Instance != 0);
  89. }
  90. //----------------------------------------------------------------------------
  91. template<typename BaseClassType>
  92. void ctkAbstractFactoryItem<BaseClassType>::uninstantiate()
  93. {
  94. if (!this->Instance)
  95. {
  96. return;
  97. }
  98. delete this->Instance;
  99. // Make sure the pointer is set to 0. Doing so, Will prevent attempt to
  100. // delete unextising object if uninstantiate() methods is called multiple times.
  101. this->Instance = 0;
  102. }
  103. //----------------------------------------------------------------------------
  104. template<typename BaseClassType>
  105. void ctkAbstractFactoryItem<BaseClassType>::setVerbose(bool value)
  106. {
  107. this->Verbose = value;
  108. }
  109. //----------------------------------------------------------------------------
  110. template<typename BaseClassType>
  111. bool ctkAbstractFactoryItem<BaseClassType>::verbose()const
  112. {
  113. return this->Verbose;
  114. }
  115. //----------------------------------------------------------------------------
  116. // ctkAbstractFactory methods
  117. //----------------------------------------------------------------------------
  118. template<typename BaseClassType>
  119. ctkAbstractFactory<BaseClassType>::ctkAbstractFactory()
  120. {
  121. this->Verbose = false;
  122. this->SharedRegisteredItemMap = QSharedPointer<HashType>(new HashType);
  123. }
  124. //----------------------------------------------------------------------------
  125. template<typename BaseClassType>
  126. ctkAbstractFactory<BaseClassType>::~ctkAbstractFactory()
  127. {
  128. }
  129. //----------------------------------------------------------------------------
  130. template<typename BaseClassType>
  131. void ctkAbstractFactory<BaseClassType>::printAdditionalInfo()
  132. {
  133. qDebug() << "ctkAbstractFactory<BaseClassType> (" << this << ")";
  134. // TODO
  135. }
  136. //----------------------------------------------------------------------------
  137. template<typename BaseClassType>
  138. BaseClassType* ctkAbstractFactory<BaseClassType>::instantiate(const QString& itemKey)
  139. {
  140. ctkAbstractFactoryItem<BaseClassType>* _item = this->item(itemKey);
  141. BaseClassType* instance = 0;
  142. bool wasInstantiated = false;
  143. if (_item)
  144. {
  145. wasInstantiated = _item->instantiated();
  146. instance = _item->instantiate();
  147. }
  148. if (!wasInstantiated)
  149. {
  150. this->displayStatusMessage(instance ? QtDebugMsg : QtCriticalMsg,
  151. QString("Attempt to instantiate \"%1\"").arg(itemKey),
  152. instance ? "OK" : "Failed", this->verbose());
  153. if (_item)
  154. {
  155. if(!_item->instantiateErrorStrings().isEmpty())
  156. {
  157. qCritical().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Error(s):\n"))
  158. << qPrintable(QString(" ").repeated(4) +
  159. _item->instantiateErrorStrings().join(
  160. QString("\n") + QString(" ").repeated(4)));
  161. }
  162. if(!_item->instantiateWarningStrings().isEmpty())
  163. {
  164. qWarning().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Warning(s):\n"))
  165. << qPrintable(QString(" ").repeated(4) +
  166. _item->instantiateWarningStrings().join(
  167. QString("\n") + QString(" ").repeated(4)));
  168. }
  169. }
  170. }
  171. return instance;
  172. }
  173. //----------------------------------------------------------------------------
  174. template<typename BaseClassType>
  175. void ctkAbstractFactory<BaseClassType>::uninstantiate(const QString& itemKey)
  176. {
  177. ctkAbstractFactoryItem<BaseClassType> * _item = this->item(itemKey);
  178. if (!_item)
  179. {
  180. return;
  181. }
  182. _item->uninstantiate();
  183. }
  184. //----------------------------------------------------------------------------
  185. template<typename BaseClassType>
  186. void ctkAbstractFactory<BaseClassType>::setSharedItems(const QSharedPointer<HashType>& items)
  187. {
  188. this->SharedRegisteredItemMap = items;
  189. }
  190. //----------------------------------------------------------------------------
  191. template<typename BaseClassType>
  192. QSharedPointer<typename ctkAbstractFactory<BaseClassType>::HashType>
  193. ctkAbstractFactory<BaseClassType>::sharedItems()
  194. {
  195. return this->SharedRegisteredItemMap;
  196. }
  197. //----------------------------------------------------------------------------
  198. template<typename BaseClassType>
  199. QStringList ctkAbstractFactory<BaseClassType>::itemKeys() const
  200. {
  201. // Since by construction, we checked if a name was already in the QHash,
  202. // there is no need to call 'uniqueKeys'
  203. return this->RegisteredItemMap.keys();
  204. }
  205. //----------------------------------------------------------------------------
  206. template<typename BaseClassType>
  207. void ctkAbstractFactory<BaseClassType>::displayStatusMessage(
  208. const QtMsgType& type, const QString& description, const QString& status, bool display)
  209. {
  210. QString msg = QString("%1 [%2]").arg(description + " ", -70, QChar('.')).arg(status);
  211. if (display)
  212. {
  213. switch(type)
  214. {
  215. case QtFatalMsg:
  216. qFatal("%s", qPrintable(msg));
  217. break;
  218. case QtCriticalMsg:
  219. qCritical("%s", qPrintable(msg));
  220. break;
  221. case QtWarningMsg:
  222. qWarning("%s", qPrintable(msg));
  223. break;
  224. case QtDebugMsg:
  225. qDebug("%s", qPrintable(msg));
  226. break;
  227. }
  228. }
  229. }
  230. //----------------------------------------------------------------------------
  231. template<typename BaseClassType>
  232. bool ctkAbstractFactory<BaseClassType>::registerItem(const QString& key,
  233. const QSharedPointer<ctkAbstractFactoryItem<BaseClassType> > & _item)
  234. {
  235. // Sanity checks
  236. if (!_item)
  237. {
  238. if (this->verbose())
  239. {
  240. qDebug() << __FUNCTION__ << "key is empty - item: " << _item;
  241. }
  242. return false;
  243. }
  244. QString description = QString("Attempt to register \"%1\"").arg(key);
  245. if (this->item(key))
  246. {
  247. this->displayStatusMessage(QtWarningMsg, description, "Already registered", this->verbose());
  248. return false;
  249. }
  250. if (this->sharedItem(key))
  251. {
  252. this->displayStatusMessage(QtDebugMsg, description,
  253. "Already registered in other factory", this->verbose());
  254. return false;
  255. }
  256. // Attempt to load it
  257. if (!_item->load())
  258. {
  259. this->displayStatusMessage(QtCriticalMsg, description, "Failed", this->verbose());
  260. if (this->verbose() && !_item->loadErrorString().isEmpty())
  261. {
  262. qCritical().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Error(s):\n"))
  263. << qPrintable(QString(" ").repeated(4) + _item->loadErrorString());
  264. }
  265. return false;
  266. }
  267. // Store item reference using a QSharedPointer
  268. this->RegisteredItemMap.insert(key, _item);
  269. this->SharedRegisteredItemMap.data()->insert(key, _item);
  270. this->displayStatusMessage(QtDebugMsg, description, "OK", this->verbose());
  271. return true;
  272. }
  273. //----------------------------------------------------------------------------
  274. template<typename BaseClassType>
  275. ctkAbstractFactoryItem<BaseClassType> * ctkAbstractFactory<BaseClassType>::item(const QString& itemKey)const
  276. {
  277. ConstIterator iter = this->RegisteredItemMap.find(itemKey);
  278. if ( iter == this->RegisteredItemMap.constEnd())
  279. {
  280. return 0;
  281. }
  282. return iter.value().data();
  283. }
  284. //----------------------------------------------------------------------------
  285. template<typename BaseClassType>
  286. ctkAbstractFactoryItem<BaseClassType> * ctkAbstractFactory<BaseClassType>::sharedItem(const QString& itemKey)const
  287. {
  288. if(this->SharedRegisteredItemMap.isNull())
  289. {
  290. return 0;
  291. }
  292. ConstIterator iter = this->SharedRegisteredItemMap.data()->find(itemKey);
  293. if ( iter == this->SharedRegisteredItemMap.data()->constEnd())
  294. {
  295. return 0;
  296. }
  297. return iter.value().data();
  298. }
  299. //----------------------------------------------------------------------------
  300. template<typename BaseClassType>
  301. void ctkAbstractFactory<BaseClassType>::setVerbose(bool value)
  302. {
  303. this->Verbose = value;
  304. }
  305. //----------------------------------------------------------------------------
  306. template<typename BaseClassType>
  307. bool ctkAbstractFactory<BaseClassType>::verbose()const
  308. {
  309. return this->Verbose;
  310. }
  311. #endif