xref: /haiku/src/kits/media/MediaFiles.cpp (revision ed24eb5ff12640d052171c6a7feba37fab8a75d1)
1 /*
2  * Copyright 2002-2009, Haiku, Inc. All rights reserved.
3  * Distributed under the terms of the MIT License.
4  *
5  * Authors:
6  *		Marcus Overhagen
7  *		Jérôme Duval
8  */
9 
10 
11 #include <MediaFiles.h>
12 
13 #include <AppMisc.h>
14 #include <DataExchange.h>
15 #include <MediaDebug.h>
16 
17 
18 const char BMediaFiles::B_SOUNDS[] = "Sounds";
19 
20 
21 BMediaFiles::BMediaFiles()
22 	:
23 	fTypeIndex(-1),
24 	fItemIndex(-1)
25 {
26 
27 }
28 
29 
30 BMediaFiles::~BMediaFiles()
31 {
32 	_ClearTypes();
33 	_ClearItems();
34 }
35 
36 
37 status_t
38 BMediaFiles::RewindTypes()
39 {
40 	CALLED();
41 
42 	_ClearTypes();
43 
44 	server_get_media_types_request request;
45 	request.team = BPrivate::current_team();
46 
47 	server_get_media_types_reply reply;
48 	status_t status = QueryServer(SERVER_GET_MEDIA_FILE_TYPES, &request,
49 		sizeof(request), &reply, sizeof(reply));
50 	if (status != B_OK) {
51 		ERROR("BMediaFiles::RewindTypes: failed to rewind types: %s\n",
52 			strerror(status));
53 		return status;
54 	}
55 
56 	const char* types = (const char*)reply.address;
57 	for (int32 i = 0; i < reply.count; i++)
58 		fTypes.AddItem(new BString(types + i * B_MEDIA_NAME_LENGTH));
59 
60 	delete_area(reply.area);
61 
62 	fTypeIndex = 0;
63 	return B_OK;
64 }
65 
66 
67 status_t
68 BMediaFiles::GetNextType(BString* _type)
69 {
70 	CALLED();
71 	if (fTypeIndex < 0 || fTypeIndex >= fTypes.CountItems()) {
72 		_ClearTypes();
73 		fTypeIndex = -1;
74 		return B_BAD_INDEX;
75 	}
76 
77 	*_type = *(BString*)fTypes.ItemAt(fTypeIndex);
78 	fTypeIndex++;
79 
80 	return B_OK;
81 }
82 
83 
84 status_t
85 BMediaFiles::RewindRefs(const char* type)
86 {
87 	CALLED();
88 
89 	_ClearItems();
90 
91 	TRACE("BMediaFiles::RewindRefs: sending SERVER_GET_MEDIA_FILE_ITEMS for "
92 		"type %s\n", type);
93 
94 	server_get_media_items_request request;
95 	request.team = BPrivate::current_team();
96 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
97 
98 	server_get_media_items_reply reply;
99 	status_t status = QueryServer(SERVER_GET_MEDIA_FILE_ITEMS, &request,
100 		sizeof(request), &reply, sizeof(reply));
101 	if (status != B_OK) {
102 		ERROR("BMediaFiles::RewindRefs: failed to rewind refs: %s\n",
103 			strerror(status));
104 		return status;
105 	}
106 
107 	const char* items = (const char*)reply.address;
108 	for (int32 i = 0; i < reply.count; i++) {
109 		fItems.AddItem(new BString(items + i * B_MEDIA_NAME_LENGTH,
110 			B_MEDIA_NAME_LENGTH));
111 	}
112 
113 	delete_area(reply.area);
114 
115 	fCurrentType = type;
116 	fItemIndex = 0;
117 	return B_OK;
118 }
119 
120 
121 status_t
122 BMediaFiles::GetNextRef(BString* _type, entry_ref* _ref)
123 {
124 	CALLED();
125 	if (fItemIndex < 0 || fItemIndex >= fItems.CountItems()) {
126 		_ClearItems();
127 		fItemIndex = -1;
128 		return B_BAD_INDEX;
129 	}
130 
131 	*_type = *(BString*)fItems.ItemAt(fItemIndex);
132 	GetRefFor(fCurrentType.String(), _type->String(), _ref);
133 
134 	fItemIndex++;
135 	return B_OK;
136 }
137 
138 
139 status_t
140 BMediaFiles::GetRefFor(const char* type, const char* item, entry_ref* _ref)
141 {
142 	CALLED();
143 
144 	if (type == NULL || item == NULL || _ref == NULL)
145 		return B_BAD_VALUE;
146 
147 	server_get_ref_for_request request;
148 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
149 	strlcpy(request.item, item, B_MEDIA_NAME_LENGTH);
150 
151 	server_get_ref_for_reply reply;
152 	status_t status = QueryServer(SERVER_GET_REF_FOR, &request, sizeof(request),
153 		&reply, sizeof(reply));
154 	if (status != B_OK) {
155 		ERROR("BMediaFiles::GetRefFor: failed: %s\n", strerror(status));
156 		return status;
157 	}
158 
159 	*_ref = reply.ref;
160 	return B_OK;
161 }
162 
163 
164 status_t
165 BMediaFiles::GetAudioGainFor(const char* type, const char* item, float* _gain)
166 {
167 	CALLED();
168 
169 	if (type == NULL || item == NULL || _gain == NULL)
170 		return B_BAD_VALUE;
171 
172 	server_get_item_audio_gain_request request;
173 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
174 	strlcpy(request.item, item, B_MEDIA_NAME_LENGTH);
175 
176 	server_get_item_audio_gain_reply reply;
177 	status_t status = QueryServer(SERVER_GET_ITEM_AUDIO_GAIN, &request,
178 		sizeof(request), &reply, sizeof(reply));
179 	if (status != B_OK) {
180 		ERROR("BMediaFiles::GetRefFor: failed: %s\n", strerror(status));
181 		return status;
182 	}
183 
184 	*_gain = reply.gain;
185 	return B_OK;
186 }
187 
188 
189 status_t
190 BMediaFiles::SetRefFor(const char* type, const char* item,
191 	const entry_ref& ref)
192 {
193 	CALLED();
194 
195 	server_set_ref_for_request request;
196 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
197 	strlcpy(request.item, item, B_MEDIA_NAME_LENGTH);
198 	request.ref = ref;
199 
200 	server_set_ref_for_reply reply;
201 	status_t status = QueryServer(SERVER_SET_REF_FOR, &request, sizeof(request),
202 		&reply, sizeof(reply));
203 	if (status != B_OK) {
204 		ERROR("BMediaFiles::SetRefFor: failed: %s\n", strerror(status));
205 		return status;
206 	}
207 
208 	return B_OK;
209 }
210 
211 
212 status_t
213 BMediaFiles::SetAudioGainFor(const char* type, const char* item, float gain)
214 {
215 	CALLED();
216 
217 	server_set_item_audio_gain_request request;
218 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
219 	strlcpy(request.item, item, B_MEDIA_NAME_LENGTH);
220 	request.gain = gain;
221 
222 	server_set_item_audio_gain_reply reply;
223 	status_t status = QueryServer(SERVER_SET_ITEM_AUDIO_GAIN, &request,
224 		sizeof(request), &reply, sizeof(reply));
225 	if (status != B_OK) {
226 		ERROR("BMediaFiles::SetAudioGainFor: failed: %s\n", strerror(status));
227 		return status;
228 	}
229 
230 	return B_OK;
231 }
232 
233 
234 status_t
235 BMediaFiles::RemoveRefFor(const char* type, const char* item,
236 	const entry_ref &ref)
237 {
238 	CALLED();
239 
240 	server_invalidate_item_request request;
241 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
242 	strlcpy(request.item, item, B_MEDIA_NAME_LENGTH);
243 
244 	server_invalidate_item_reply reply;
245 	status_t status = QueryServer(SERVER_INVALIDATE_MEDIA_ITEM, &request,
246 		sizeof(request), &reply, sizeof(reply));
247 	if (status != B_OK) {
248 		ERROR("BMediaFiles::RemoveRefFor: failed: %s\n", strerror(status));
249 		return status;
250 	}
251 
252 	return B_OK;
253 }
254 
255 
256 status_t
257 BMediaFiles::RemoveItem(const char* type, const char* item)
258 {
259 	CALLED();
260 
261 	server_remove_media_item_request request;
262 	strlcpy(request.type, type, B_MEDIA_NAME_LENGTH);
263 	strlcpy(request.item, item, B_MEDIA_NAME_LENGTH);
264 
265 	server_remove_media_item_reply reply;
266 	status_t status = QueryServer(SERVER_REMOVE_MEDIA_ITEM, &request,
267 		sizeof(request), &reply, sizeof(reply));
268 	if (status != B_OK) {
269 		ERROR("BMediaFiles::RemoveItem: failed: %s\n", strerror(status));
270 		return status;
271 	}
272 
273 	return B_OK;
274 }
275 
276 
277 // #pragma mark - private
278 
279 
280 void
281 BMediaFiles::_ClearTypes()
282 {
283 	for (int32 i = 0; i < fTypes.CountItems(); i++)
284 		delete (BString*)fTypes.ItemAt(i);
285 
286 	fTypes.MakeEmpty();
287 }
288 
289 
290 void
291 BMediaFiles::_ClearItems()
292 {
293 	for (int32 i = 0; i < fItems.CountItems(); i++)
294 		delete (BString*)fItems.ItemAt(i);
295 
296 	fItems.MakeEmpty();
297 }
298 
299 
300 // #pragma mark - FBC padding
301 
302 
303 status_t
304 BMediaFiles::_Reserved_MediaFiles_0(void*,...)
305 {
306 	// TODO: Someone didn't understand FBC
307 	return B_ERROR;
308 }
309 
310 
311 status_t BMediaFiles::_Reserved_MediaFiles_1(void*,...) { return B_ERROR; }
312 status_t BMediaFiles::_Reserved_MediaFiles_2(void*,...) { return B_ERROR; }
313 status_t BMediaFiles::_Reserved_MediaFiles_3(void*,...) { return B_ERROR; }
314 status_t BMediaFiles::_Reserved_MediaFiles_4(void*,...) { return B_ERROR; }
315 status_t BMediaFiles::_Reserved_MediaFiles_5(void*,...) { return B_ERROR; }
316 status_t BMediaFiles::_Reserved_MediaFiles_6(void*,...) { return B_ERROR; }
317 status_t BMediaFiles::_Reserved_MediaFiles_7(void*,...) { return B_ERROR; }
318 
319