Crazy Eddie's GUI System  0.8.4
 All Classes Namespaces Functions Variables Typedefs Enumerations Enumerator Friends Pages
NamedXMLResourceManager.h
1 /***********************************************************************
2  created: Fri Jul 17 2009
3  author: Paul D Turner <paul@cegui.org.uk>
4 *************************************************************************/
5 /***************************************************************************
6  * Copyright (C) 2004 - 2009 Paul D Turner & The CEGUI Development Team
7  *
8  * Permission is hereby granted, free of charge, to any person obtaining
9  * a copy of this software and associated documentation files (the
10  * "Software"), to deal in the Software without restriction, including
11  * without limitation the rights to use, copy, modify, merge, publish,
12  * distribute, sublicense, and/or sell copies of the Software, and to
13  * permit persons to whom the Software is furnished to do so, subject to
14  * the following conditions:
15  *
16  * The above copyright notice and this permission notice shall be
17  * included in all copies or substantial portions of the Software.
18  *
19  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
20  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
22  * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
23  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
24  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
25  * OTHER DEALINGS IN THE SOFTWARE.
26  ***************************************************************************/
27 #ifndef _CEGUINamedXMLResourceManager_h_
28 #define _CEGUINamedXMLResourceManager_h_
29 
30 #include "CEGUI/EventSet.h"
31 #include "CEGUI/String.h"
32 #include "CEGUI/Exceptions.h"
33 #include "CEGUI/Logger.h"
34 #include "CEGUI/InputEvent.h"
35 #include "CEGUI/System.h"
36 #include <map>
37 
38 // Start of CEGUI namespace section
39 namespace CEGUI
40 {
43 {
50 };
51 
52 //----------------------------------------------------------------------------//
53 
55 class CEGUIEXPORT ResourceEventSet : public EventSet
56 {
57 public:
59  static const String EventNamespace;
81 };
82 
83 //----------------------------------------------------------------------------//
84 
107 template<typename T, typename U>
109 {
110 public:
120  NamedXMLResourceManager(const String& resource_type);
121 
123  virtual ~NamedXMLResourceManager();
124 
142  T& createFromContainer(const RawDataContainer& source,
144 
167  T& createFromFile(const String& xml_filename, const String& resource_group = "",
169 
187  T& createFromString(const String& source,
189 
198  void destroy(const String& object_name);
199 
209  void destroy(const T& object);
210 
212  void destroyAll();
213 
224  T& get(const String& object_name) const;
225 
227  bool isDefined(const String& object_name) const;
228 
230  void createAll(const String& pattern, const String& resource_group);
231 
232 protected:
233  // singleton allocator fits here, resource managers are very likely to be singletons
235  typedef std::map<String, T*, StringFastLessCompare
236  CEGUI_MAP_ALLOC(String, T*)> ObjectRegistry;
238  void destroyObject(typename ObjectRegistry::iterator ob);
240  T& doExistingObjectAction(const String object_name, T* object,
241  const XMLResourceExistsAction action);
243  virtual void doPostObjectAdditionAction(T& object);
248 };
249 
250 //----------------------------------------------------------------------------//
251 template<typename T, typename U>
253  const String& resource_type) :
254  d_resourceType(resource_type)
255 {
256 }
257 
258 //----------------------------------------------------------------------------//
259 template<typename T, typename U>
261 {
262 }
263 
264 //----------------------------------------------------------------------------//
265 template<typename T, typename U>
268 {
269  U xml_loader;
270 
271  xml_loader.handleContainer(source);
272  return doExistingObjectAction(xml_loader.getObjectName(),
273  &xml_loader.getObject(), action);
274 }
275 
276 //----------------------------------------------------------------------------//
277 template<typename T, typename U>
279  const String& resource_group,
281 {
282  U xml_loader;
283 
284  xml_loader.handleFile(xml_filename, resource_group);
285  return doExistingObjectAction(xml_loader.getObjectName(),
286  &xml_loader.getObject(), action);
287 }
288 
289 //----------------------------------------------------------------------------//
290 template<typename T, typename U>
293 {
294  U xml_loader;
295 
296  xml_loader.handleString(source);
297  return doExistingObjectAction(xml_loader.getObjectName(),
298  &xml_loader.getObject(), action);
299 }
300 
301 //----------------------------------------------------------------------------//
302 template<typename T, typename U>
304 {
305  typename ObjectRegistry::iterator i(d_objects.find(object_name));
306 
307  // exit if no such object.
308  if (i == d_objects.end())
309  return;
310 
311  destroyObject(i);
312 }
313 
314 //----------------------------------------------------------------------------//
315 template<typename T, typename U>
317 {
318  // don't want to force a 'getName' function on T here, so we'll look for the
319  // object the hard way.
320  typename ObjectRegistry::iterator i(d_objects.begin());
321  for (; i != d_objects.end(); ++i)
322  if (i->second == &object)
323  {
324  destroyObject(i);
325  return;
326  }
327 }
328 
329 //----------------------------------------------------------------------------//
330 template<typename T, typename U>
332 {
333  while (!d_objects.empty())
334  destroyObject(d_objects.begin());
335 }
336 
337 //----------------------------------------------------------------------------//
338 template<typename T, typename U>
339 T& NamedXMLResourceManager<T, U>::get(const String& object_name) const
340 {
341  typename ObjectRegistry::const_iterator i(d_objects.find(object_name));
342 
343  if (i == d_objects.end())
344  CEGUI_THROW(UnknownObjectException(
345  "No object of type '" + d_resourceType + "' named '" + object_name +
346  "' is present in the collection."));
347 
348  return *i->second;
349 }
350 
351 //----------------------------------------------------------------------------//
352 template<typename T, typename U>
353 bool NamedXMLResourceManager<T, U>::isDefined(const String& object_name) const
354 {
355  return d_objects.find(object_name) != d_objects.end();
356 }
357 
358 //----------------------------------------------------------------------------//
359 template<typename T, typename U>
361  typename ObjectRegistry::iterator ob)
362 {
363  char addr_buff[32];
364  sprintf(addr_buff, "(%p)", static_cast<void*>(ob->second));
365  Logger::getSingleton().logEvent("Object of type '" + d_resourceType +
366  "' named '" + ob->first + "' has been destroyed. " +
367  addr_buff, Informative);
368 
369  // Set up event args for event notification
370  ResourceEventArgs args(d_resourceType, ob->first);
371 
372  CEGUI_DELETE_AO ob->second;
373  d_objects.erase(ob);
374 
375  // fire event signalling an object has been destroyed
376  fireEvent(EventResourceDestroyed, args, EventNamespace);
377 }
378 
379 //----------------------------------------------------------------------------//
380 template<typename T, typename U>
382  const String object_name,
383  T* object,
384  const XMLResourceExistsAction action)
385 {
386  String event_name;
387 
388  if (isDefined(object_name))
389  {
390  switch (action)
391  {
392  case XREA_RETURN:
393  Logger::getSingleton().logEvent("---- Returning existing instance "
394  "of " + d_resourceType + " named '" + object_name + "'.");
395  // delete any new object we already had created
396  CEGUI_DELETE_AO object;
397  // return existing instance of object.
398  return *d_objects[object_name];
399 
400  case XREA_REPLACE:
401  Logger::getSingleton().logEvent("---- Replacing existing instance "
402  "of " + d_resourceType + " named '" + object_name +
403  "' (DANGER!).");
404  destroy(object_name);
405  event_name = EventResourceReplaced;
406  break;
407 
408  case XREA_THROW:
409  CEGUI_DELETE_AO object;
410  CEGUI_THROW(AlreadyExistsException(
411  "an object of type '" + d_resourceType + "' named '" +
412  object_name + "' already exists in the collection."));
413 
414  default:
415  CEGUI_DELETE_AO object;
416  CEGUI_THROW(InvalidRequestException(
417  "Invalid CEGUI::XMLResourceExistsAction was specified."));
418  }
419  }
420  else
421  event_name = EventResourceCreated;
422 
423  d_objects[object_name] = object;
424  doPostObjectAdditionAction(*object);
425 
426  // fire event about this resource change
427  ResourceEventArgs args(d_resourceType, object_name);
428  fireEvent(event_name, args, EventNamespace);
429 
430  return *object;
431 }
432 
433 //----------------------------------------------------------------------------//
434 template<typename T, typename U>
436 {
437  // do nothing by default.
438 }
439 
440 //----------------------------------------------------------------------------//
441 template<typename T, typename U>
443  const String& resource_group)
444 {
445  std::vector<String> names;
446  const size_t num = System::getSingleton().getResourceProvider()->
447  getResourceGroupFileNames(names, pattern, resource_group);
448 
449  for (size_t i = 0; i < num; ++i)
450  createFromFile(names[i], resource_group);
451 }
452 
453 //----------------------------------------------------------------------------//
454 
455 } // End of CEGUI namespace section
456 
457 #endif // end of guard _CEGUINamedXMLResourceManager_h_