ctkAbstractFactory.tpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401
  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. QStringList ctkAbstractFactoryItem<BaseClassType>::instantiateErrorStrings()const
  32. {
  33. return this->InstantiateErrorStrings;
  34. }
  35. //----------------------------------------------------------------------------
  36. template<typename BaseClassType>
  37. void ctkAbstractFactoryItem<BaseClassType>::appendInstantiateErrorString(const QString& errorString)
  38. {
  39. this->InstantiateErrorStrings << errorString;
  40. }
  41. //----------------------------------------------------------------------------
  42. template<typename BaseClassType>
  43. void ctkAbstractFactoryItem<BaseClassType>::clearInstantiateErrorStrings()
  44. {
  45. this->InstantiateErrorStrings.clear();
  46. }
  47. //----------------------------------------------------------------------------
  48. template<typename BaseClassType>
  49. QStringList ctkAbstractFactoryItem<BaseClassType>::instantiateWarningStrings()const
  50. {
  51. return this->InstantiateWarningStrings;
  52. }
  53. //----------------------------------------------------------------------------
  54. template<typename BaseClassType>
  55. void ctkAbstractFactoryItem<BaseClassType>::appendInstantiateWarningString(const QString& msg)
  56. {
  57. this->InstantiateWarningStrings << msg;
  58. }
  59. //----------------------------------------------------------------------------
  60. template<typename BaseClassType>
  61. void ctkAbstractFactoryItem<BaseClassType>::clearInstantiateWarningStrings()
  62. {
  63. this->InstantiateWarningStrings.clear();
  64. }
  65. //----------------------------------------------------------------------------
  66. template<typename BaseClassType>
  67. QStringList ctkAbstractFactoryItem<BaseClassType>::loadErrorStrings()const
  68. {
  69. return this->LoadErrorStrings;
  70. }
  71. //----------------------------------------------------------------------------
  72. template<typename BaseClassType>
  73. void ctkAbstractFactoryItem<BaseClassType>::appendLoadErrorString(const QString& errorString)
  74. {
  75. this->LoadErrorStrings << errorString;
  76. }
  77. //----------------------------------------------------------------------------
  78. template<typename BaseClassType>
  79. void ctkAbstractFactoryItem<BaseClassType>::clearLoadErrorStrings()
  80. {
  81. this->LoadErrorStrings.clear();
  82. }
  83. //----------------------------------------------------------------------------
  84. template<typename BaseClassType>
  85. QStringList ctkAbstractFactoryItem<BaseClassType>::loadWarningStrings()const
  86. {
  87. return this->LoadWarningStrings;
  88. }
  89. //----------------------------------------------------------------------------
  90. template<typename BaseClassType>
  91. void ctkAbstractFactoryItem<BaseClassType>::appendLoadWarningString(const QString& msg)
  92. {
  93. this->LoadWarningStrings << msg;
  94. }
  95. //----------------------------------------------------------------------------
  96. template<typename BaseClassType>
  97. void ctkAbstractFactoryItem<BaseClassType>::clearLoadWarningStrings()
  98. {
  99. this->LoadWarningStrings.clear();
  100. }
  101. //----------------------------------------------------------------------------
  102. template<typename BaseClassType>
  103. BaseClassType* ctkAbstractFactoryItem<BaseClassType>::instantiate()
  104. {
  105. this->clearInstantiateErrorStrings();
  106. this->clearInstantiateWarningStrings();
  107. if (this->Instance)
  108. {
  109. return this->Instance;
  110. }
  111. this->Instance = this->instanciator();
  112. return this->Instance;
  113. }
  114. //----------------------------------------------------------------------------
  115. template<typename BaseClassType>
  116. bool ctkAbstractFactoryItem<BaseClassType>::instantiated()const
  117. {
  118. return (this->Instance != 0);
  119. }
  120. //----------------------------------------------------------------------------
  121. template<typename BaseClassType>
  122. void ctkAbstractFactoryItem<BaseClassType>::uninstantiate()
  123. {
  124. if (!this->Instance)
  125. {
  126. return;
  127. }
  128. delete this->Instance;
  129. // Make sure the pointer is set to 0. Doing so, Will prevent attempt to
  130. // delete unextising object if uninstantiate() methods is called multiple times.
  131. this->Instance = 0;
  132. }
  133. //----------------------------------------------------------------------------
  134. template<typename BaseClassType>
  135. void ctkAbstractFactoryItem<BaseClassType>::setVerbose(bool value)
  136. {
  137. this->Verbose = value;
  138. }
  139. //----------------------------------------------------------------------------
  140. template<typename BaseClassType>
  141. bool ctkAbstractFactoryItem<BaseClassType>::verbose()const
  142. {
  143. return this->Verbose;
  144. }
  145. //----------------------------------------------------------------------------
  146. // ctkAbstractFactory methods
  147. //----------------------------------------------------------------------------
  148. template<typename BaseClassType>
  149. ctkAbstractFactory<BaseClassType>::ctkAbstractFactory()
  150. {
  151. this->Verbose = false;
  152. this->SharedRegisteredItemMap = QSharedPointer<HashType>(new HashType);
  153. }
  154. //----------------------------------------------------------------------------
  155. template<typename BaseClassType>
  156. ctkAbstractFactory<BaseClassType>::~ctkAbstractFactory()
  157. {
  158. }
  159. //----------------------------------------------------------------------------
  160. template<typename BaseClassType>
  161. void ctkAbstractFactory<BaseClassType>::printAdditionalInfo()
  162. {
  163. qDebug() << "ctkAbstractFactory<BaseClassType> (" << this << ")";
  164. // TODO
  165. }
  166. //----------------------------------------------------------------------------
  167. template<typename BaseClassType>
  168. BaseClassType* ctkAbstractFactory<BaseClassType>::instantiate(const QString& itemKey)
  169. {
  170. ctkAbstractFactoryItem<BaseClassType>* _item = this->item(itemKey);
  171. BaseClassType* instance = 0;
  172. bool wasInstantiated = false;
  173. if (_item)
  174. {
  175. wasInstantiated = _item->instantiated();
  176. instance = _item->instantiate();
  177. }
  178. if (!wasInstantiated)
  179. {
  180. this->displayStatusMessage(instance ? QtDebugMsg : QtCriticalMsg,
  181. QString("Attempt to instantiate \"%1\"").arg(itemKey),
  182. instance ? "OK" : "Failed", this->verbose());
  183. if (_item)
  184. {
  185. if(!_item->instantiateErrorStrings().isEmpty())
  186. {
  187. qCritical().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Error(s):\n"))
  188. << qPrintable(QString(" ").repeated(4) +
  189. _item->instantiateErrorStrings().join(
  190. QString("\n") + QString(" ").repeated(4)));
  191. }
  192. if(!_item->instantiateWarningStrings().isEmpty())
  193. {
  194. qWarning().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Warning(s):\n"))
  195. << qPrintable(QString(" ").repeated(4) +
  196. _item->instantiateWarningStrings().join(
  197. QString("\n") + QString(" ").repeated(4)));
  198. }
  199. }
  200. }
  201. return instance;
  202. }
  203. //----------------------------------------------------------------------------
  204. template<typename BaseClassType>
  205. void ctkAbstractFactory<BaseClassType>::uninstantiate(const QString& itemKey)
  206. {
  207. ctkAbstractFactoryItem<BaseClassType> * _item = this->item(itemKey);
  208. if (!_item)
  209. {
  210. return;
  211. }
  212. _item->uninstantiate();
  213. }
  214. //----------------------------------------------------------------------------
  215. template<typename BaseClassType>
  216. void ctkAbstractFactory<BaseClassType>::setSharedItems(const QSharedPointer<HashType>& items)
  217. {
  218. this->SharedRegisteredItemMap = items;
  219. }
  220. //----------------------------------------------------------------------------
  221. template<typename BaseClassType>
  222. QSharedPointer<typename ctkAbstractFactory<BaseClassType>::HashType>
  223. ctkAbstractFactory<BaseClassType>::sharedItems()
  224. {
  225. return this->SharedRegisteredItemMap;
  226. }
  227. //----------------------------------------------------------------------------
  228. template<typename BaseClassType>
  229. QStringList ctkAbstractFactory<BaseClassType>::itemKeys() const
  230. {
  231. // Since by construction, we checked if a name was already in the QHash,
  232. // there is no need to call 'uniqueKeys'
  233. return this->RegisteredItemMap.keys();
  234. }
  235. //----------------------------------------------------------------------------
  236. template<typename BaseClassType>
  237. void ctkAbstractFactory<BaseClassType>::displayStatusMessage(
  238. const QtMsgType& type, const QString& description, const QString& status, bool display)
  239. {
  240. QString msg = QString("%1 [%2]").arg(description + " ", -70, QChar('.')).arg(status);
  241. if (display)
  242. {
  243. switch(type)
  244. {
  245. case QtFatalMsg:
  246. qFatal("%s", qPrintable(msg));
  247. break;
  248. case QtCriticalMsg:
  249. qCritical("%s", qPrintable(msg));
  250. break;
  251. case QtWarningMsg:
  252. qWarning("%s", qPrintable(msg));
  253. break;
  254. case QtDebugMsg:
  255. qDebug("%s", qPrintable(msg));
  256. break;
  257. }
  258. }
  259. }
  260. //----------------------------------------------------------------------------
  261. template<typename BaseClassType>
  262. bool ctkAbstractFactory<BaseClassType>::registerItem(const QString& key,
  263. const QSharedPointer<ctkAbstractFactoryItem<BaseClassType> > & _item)
  264. {
  265. // Sanity checks
  266. if (!_item)
  267. {
  268. if (this->verbose())
  269. {
  270. qDebug() << __FUNCTION__ << "key is empty - item: " << _item;
  271. }
  272. return false;
  273. }
  274. QString description = QString("Attempt to register \"%1\"").arg(key);
  275. if (this->item(key))
  276. {
  277. this->displayStatusMessage(QtWarningMsg, description, "Already registered", this->verbose());
  278. return false;
  279. }
  280. if (this->sharedItem(key))
  281. {
  282. this->displayStatusMessage(QtDebugMsg, description,
  283. "Already registered in other factory", this->verbose());
  284. return false;
  285. }
  286. // Attempt to load it
  287. if (!_item->load())
  288. {
  289. this->displayStatusMessage(QtCriticalMsg, description, "Failed", this->verbose());
  290. if(!_item->loadErrorStrings().isEmpty())
  291. {
  292. qCritical().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Error(s):\n"))
  293. << qPrintable(QString(" ").repeated(4) +
  294. _item->loadErrorStrings().join(
  295. QString("\n") + QString(" ").repeated(4)));
  296. }
  297. if(!_item->loadWarningStrings().isEmpty())
  298. {
  299. qWarning().nospace() << qPrintable(QString(" ").repeated(2) + QLatin1String("Warning(s):\n"))
  300. << qPrintable(QString(" ").repeated(4) +
  301. _item->loadWarningStrings().join(
  302. QString("\n") + QString(" ").repeated(4)));
  303. }
  304. return false;
  305. }
  306. // Store item reference using a QSharedPointer
  307. this->RegisteredItemMap.insert(key, _item);
  308. this->SharedRegisteredItemMap.data()->insert(key, _item);
  309. this->displayStatusMessage(QtDebugMsg, description, "OK", this->verbose());
  310. return true;
  311. }
  312. //----------------------------------------------------------------------------
  313. template<typename BaseClassType>
  314. ctkAbstractFactoryItem<BaseClassType> * ctkAbstractFactory<BaseClassType>::item(const QString& itemKey)const
  315. {
  316. ConstIterator iter = this->RegisteredItemMap.find(itemKey);
  317. if ( iter == this->RegisteredItemMap.constEnd())
  318. {
  319. return 0;
  320. }
  321. return iter.value().data();
  322. }
  323. //----------------------------------------------------------------------------
  324. template<typename BaseClassType>
  325. ctkAbstractFactoryItem<BaseClassType> * ctkAbstractFactory<BaseClassType>::sharedItem(const QString& itemKey)const
  326. {
  327. if(this->SharedRegisteredItemMap.isNull())
  328. {
  329. return 0;
  330. }
  331. ConstIterator iter = this->SharedRegisteredItemMap.data()->find(itemKey);
  332. if ( iter == this->SharedRegisteredItemMap.data()->constEnd())
  333. {
  334. return 0;
  335. }
  336. return iter.value().data();
  337. }
  338. //----------------------------------------------------------------------------
  339. template<typename BaseClassType>
  340. void ctkAbstractFactory<BaseClassType>::setVerbose(bool value)
  341. {
  342. this->Verbose = value;
  343. }
  344. //----------------------------------------------------------------------------
  345. template<typename BaseClassType>
  346. bool ctkAbstractFactory<BaseClassType>::verbose()const
  347. {
  348. return this->Verbose;
  349. }
  350. #endif