File: adding_resources.xml

package info (click to toggle)
clanlib 0.8.1-1
  • links: PTS, VCS
  • area: main
  • in suites: lenny
  • size: 28,372 kB
  • ctags: 16,520
  • sloc: cpp: 101,145; sh: 8,752; xml: 6,410; makefile: 1,740; ansic: 463; perl: 424; php: 247
file content (141 lines) | stat: -rw-r--r-- 4,557 bytes parent folder | download | duplicates (7)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
<xml>
<head>
<title>Custom Resource Types</title>
</head>
<body>

<p>Extending the resource management system to incorporate a new resource type can be tricky,
but with a little guidance, it is fairly straight forward.  </p>

<h3>Intoduction</h3>
<p>For this tutorial we will be using a simple class called Object</p>

<code>
class Object
{
public:
	Object();
	Object(int x, int y);
	int x;
	int y;
};
</code>

<h3>Modifing the class</h3>
<p>First you will need to add a constructor that will load from the resource management system and an instance of <codelink>CL_Resource</codelink> to store our resource information in.</p>

<code>
class Object
{
public:
	Object();
	Object(int x, int y);
	Object(const std::string &resource_id, CL_ResourceManager *manager);
	int x;
	int y;

private:
	CL_Resource resource;
};
</code>

<h3>Creating your resourcedata class</h3>
<p>Now, before we actually define our new constructor for Object, lets create our new <codelinke> CL_ResourceData</codelink> derivative class that will actually do the loading. This class is also fairly simple; it needs a constructor that takes a <codelink>CL_Resource</codelink> and two functions: on_load() and on_unload().  These functions will be called by the resource manager when the resource is loaded and unloaded respectively.</p>

<code>
class CL_ResourceData_Object : public CL_ResourceData
{
public:
	ResourceData_Object(CL_Resource &resource);
	virtual ~ResourceData_Object();
	Object &get_object();
private:
	void on_load();
	void on_unload();
	Object object;
};
</code>

<p>The constructor tells the resource to attach the data through this instance of CL_ResourceData.  Later when you ask your resource manager to load the object it will call the on_load() function which actually reads in your data.  Notice that the call to get_attribute takes two parameters in this instance. The first is obviously the variable name, but the second is the default value which is used when the attribute is not found.</p>
<code>
ResourceData_Object::ResourceData_Object(CL_Resource &resource)
: CL_ResourceData(resource)
{
	resource.attach_data("object", this);
}
ResourceData_Object::on_load()
{
	CL_Resource resource = get_resource();
	int x = atoi( (resource.get_element().get_attribute("x","50")).c_str() );
	int y = atoi( (resource.get_element().get_attribute("y","75")).c_str() );
 
	object.x = x;
	object.y = y;
}
ResourceData_Object::on_unload()
{
	object = Object();
}
</code>

<h3>The callback function</h3>
<p>In order for this to work, we have to create a custom callback function to load your new resource types.  The callback function just looks to see if the new resource is of our custom type, and in that event creates a new instance of our resourcedata type.</p>

<code>
static void resource_added(CL_Resource &resource)
{
	std::string type = resource.get_type();

	if (type == "object") new ResourceData_Object(resource);
}
</code>
<p>And then in your initialization routines just add a line like:</p>
<code>
CL_Slot slot_resource_added - CL_ResourceManager::sig_resource_added().connect(&resource_added);
</code>

<h3>Adding object's constructor</h3>
<p>Now just add that constructor for Object that we nearly forgot about.  This function will just ask the our <codelink>CL_ResourceManager</codelink> for the instance of our object, then we tell it load it up.  Then just get our our data from the resource, but if it returns no data then the name you specified is not of type Object.  Then just get out your data however you like, a copy constuctor or overloaded = operator comes in real handy here.</p>

<code>
Object::Object(const std::string &resource_id, CL_ResourceManager *manager)
{
	resource = manager->get_resource(resource_id);
	resource.load();

	ResourceData_Object *data =
		(ResourceData_Object *) resource.get_data("object");

	if(!data)
		throw CL_Error("Resource '" + resource_id + "' is not of type 'object'");

	x = data->get_object().x;
	y = data->get_object().y;

	resource.unload();
}
</code>

<h3>Finale</h3>
<p>The last thing to do is to call your resource. For a resource file looking like:</p>

<code>
&lt;resources&gt;
	&lt;object name="object01" x="20" y="20" /&gt;
&lt;/resources&gt;
</code>
<p>Just create your object like this:</p>

<code>
Object *my_object = new Object("object01", new CL_ResourceManager("my_resourcefile.xml"));
</code>

Thats it!

<h3>TODO:</h3>
<list>
<li>Explain how to use the resource manager's provider to load files.</li>
<li>Explain how to make the resource manager cache objects.</li>
</list>
</body>
</xml>