ctkAbstractFactory.tpp 14 KB

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