1 /*****************************************************************************/
2 // TranslatorSettings
3 // Written by Michael Wilber, Haiku Translation Kit Team
4 //
5 // TranslatorSettings.cpp
6 //
7 // This class manages (saves/loads/locks/unlocks) the settings
8 // for a Translator.
9 //
10 //
11 // Copyright (c) 2004 Haiku Project
12 //
13 // Permission is hereby granted, free of charge, to any person obtaining a
14 // copy of this software and associated documentation files (the "Software"),
15 // to deal in the Software without restriction, including without limitation
16 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
17 // and/or sell copies of the Software, and to permit persons to whom the
18 // Software is furnished to do so, subject to the following conditions:
19 //
20 // The above copyright notice and this permission notice shall be included
21 // in all copies or substantial portions of the Software.
22 //
23 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
24 // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
25 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
26 // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
27 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
28 // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
29 // DEALINGS IN THE SOFTWARE.
30 /*****************************************************************************/
31
32 #include <string.h>
33 #include <File.h>
34 #include <FindDirectory.h>
35 #include <TranslatorFormats.h>
36 // for B_TRANSLATOR_EXT_*
37 #include "TranslatorSettings.h"
38
39 // ---------------------------------------------------------------
40 // Constructor
41 //
42 // Sets the default settings, location for the settings file
43 // and sets the reference count to 1
44 //
45 // Preconditions:
46 //
47 // Parameters:
48 //
49 // Postconditions:
50 //
51 // Returns:
52 // ---------------------------------------------------------------
TranslatorSettings(const char * settingsFile,const TranSetting * defaults,int32 defCount)53 TranslatorSettings::TranslatorSettings(const char *settingsFile,
54 const TranSetting *defaults, int32 defCount)
55 : fLock("TranslatorSettings Lock")
56 {
57 if (find_directory(B_USER_SETTINGS_DIRECTORY, &fSettingsPath))
58 fSettingsPath.SetTo("/tmp");
59 fSettingsPath.Append(settingsFile);
60
61 fRefCount = 1;
62
63 if (defCount > 0) {
64 fDefaults = defaults;
65 fDefCount = defCount;
66 } else {
67 fDefaults = NULL;
68 fDefCount = 0;
69 }
70
71 // Add Default Settings
72 // (Used when loading from the settings file or from
73 // a BMessage fails)
74 const TranSetting *defs = fDefaults;
75 for (int32 i = 0; i < fDefCount; i++) {
76 switch (defs[i].dataType) {
77 case TRAN_SETTING_BOOL:
78 fSettingsMsg.AddBool(defs[i].name,
79 static_cast<bool>(defs[i].defaultVal));
80 break;
81
82 case TRAN_SETTING_INT32:
83 fSettingsMsg.AddInt32(defs[i].name, defs[i].defaultVal);
84 break;
85
86 default:
87 // ASSERT here? Erase the bogus setting entry instead?
88 break;
89 }
90 }
91 }
92
93 // ---------------------------------------------------------------
94 // Acquire
95 //
96 // Returns a pointer to the TranslatorSettings and increments
97 // the reference count.
98 //
99 // Preconditions:
100 //
101 // Parameters:
102 //
103 // Postconditions:
104 //
105 // Returns: pointer to this TranslatorSettings object
106 // ---------------------------------------------------------------
107 TranslatorSettings *
Acquire()108 TranslatorSettings::Acquire()
109 {
110 TranslatorSettings *psettings = NULL;
111
112 fLock.Lock();
113 fRefCount++;
114 psettings = this;
115 fLock.Unlock();
116
117 return psettings;
118 }
119
120 // ---------------------------------------------------------------
121 // Release
122 //
123 // Decrements the reference count and deletes the
124 // TranslatorSettings if the reference count is zero.
125 //
126 // Preconditions:
127 //
128 // Parameters:
129 //
130 // Postconditions:
131 //
132 // Returns: pointer to this TranslatorSettings object if
133 // the reference count is greater than zero, returns NULL
134 // if the reference count is zero and the TranslatorSettings
135 // object has been deleted
136 // ---------------------------------------------------------------
137 TranslatorSettings *
Release()138 TranslatorSettings::Release()
139 {
140 TranslatorSettings *psettings = NULL;
141
142 fLock.Lock();
143 fRefCount--;
144 if (fRefCount > 0) {
145 psettings = this;
146 fLock.Unlock();
147 } else
148 delete this;
149 // delete this object and
150 // release locks
151
152 return psettings;
153 }
154
155 // ---------------------------------------------------------------
156 // Destructor
157 //
158 // Does nothing!
159 //
160 // Preconditions:
161 //
162 // Parameters:
163 //
164 // Postconditions:
165 //
166 // Returns:
167 // ---------------------------------------------------------------
~TranslatorSettings()168 TranslatorSettings::~TranslatorSettings()
169 {
170 }
171
172 // ---------------------------------------------------------------
173 // LoadSettings
174 //
175 // Loads the settings by reading them from the default
176 // settings file.
177 //
178 // Preconditions:
179 //
180 // Parameters:
181 //
182 // Postconditions:
183 //
184 // Returns: B_OK if there were no errors or an error code from
185 // BFile::SetTo() or BMessage::Unflatten() if there were errors
186 // ---------------------------------------------------------------
187 status_t
LoadSettings()188 TranslatorSettings::LoadSettings()
189 {
190 status_t result;
191
192 fLock.Lock();
193
194 // Don't try to open the settings file if there are
195 // no settings that need to be loaded
196 if (fDefCount > 0) {
197 BFile settingsFile;
198 result = settingsFile.SetTo(fSettingsPath.Path(), B_READ_ONLY);
199 if (result == B_OK) {
200 BMessage msg;
201 result = msg.Unflatten(&settingsFile);
202 if (result == B_OK)
203 result = LoadSettings(&msg);
204 }
205 } else
206 result = B_OK;
207
208 fLock.Unlock();
209
210 return result;
211 }
212
213 // ---------------------------------------------------------------
214 // LoadSettings
215 //
216 // Loads the settings from a BMessage passed to the function.
217 //
218 // Preconditions:
219 //
220 // Parameters: pmsg pointer to BMessage that contains the
221 // settings
222 //
223 // Postconditions:
224 //
225 // Returns: B_BAD_VALUE if pmsg is NULL or invalid options
226 // have been found, B_OK if there were no
227 // errors or an error code from BMessage::FindBool() or
228 // BMessage::ReplaceBool() if there were other errors
229 // ---------------------------------------------------------------
230 status_t
LoadSettings(BMessage * pmsg)231 TranslatorSettings::LoadSettings(BMessage *pmsg)
232 {
233 if (pmsg == NULL)
234 return B_BAD_VALUE;
235
236 fLock.Lock();
237 const TranSetting *defaults = fDefaults;
238 for (int32 i = 0; i < fDefCount; i++) {
239 switch (defaults[i].dataType) {
240 case TRAN_SETTING_BOOL:
241 {
242 bool value;
243 if (pmsg->FindBool(defaults[i].name, &value) != B_OK) {
244 if (fSettingsMsg.HasBool(defaults[i].name))
245 break;
246 else
247 value = static_cast<bool>(defaults[i].defaultVal);
248 }
249
250 fSettingsMsg.ReplaceBool(defaults[i].name, value);
251 break;
252 }
253
254 case TRAN_SETTING_INT32:
255 {
256 int32 value;
257 if (pmsg->FindInt32(defaults[i].name, &value) != B_OK) {
258 if (fSettingsMsg.HasInt32(defaults[i].name))
259 break;
260 else
261 value = defaults[i].defaultVal;
262 }
263
264 fSettingsMsg.ReplaceInt32(defaults[i].name, value);
265 break;
266 }
267
268 default:
269 // TODO: ASSERT here? Erase the bogus setting entry instead?
270 break;
271 }
272 }
273
274 fLock.Unlock();
275 return B_OK;
276 }
277
278 // ---------------------------------------------------------------
279 // SaveSettings
280 //
281 // Saves the settings as a flattened BMessage to the default
282 // settings file
283 //
284 // Preconditions:
285 //
286 // Parameters:
287 //
288 // Postconditions:
289 //
290 // Returns: B_OK if no errors or an error code from BFile::SetTo()
291 // or BMessage::Flatten() if there were errors
292 // ---------------------------------------------------------------
293 status_t
SaveSettings()294 TranslatorSettings::SaveSettings()
295 {
296 status_t result;
297
298 fLock.Lock();
299
300 // Only write out settings file if there are
301 // actual settings stored by this object
302 if (fDefCount > 0) {
303 BFile settingsFile;
304 result = settingsFile.SetTo(fSettingsPath.Path(),
305 B_WRITE_ONLY | B_CREATE_FILE | B_ERASE_FILE);
306 if (result == B_OK)
307 result = fSettingsMsg.Flatten(&settingsFile);
308 } else
309 result = B_OK;
310
311 fLock.Unlock();
312
313 return result;
314 }
315
316 // ---------------------------------------------------------------
317 // GetConfigurationMessage
318 //
319 // Saves the current settings to the BMessage passed to the
320 // function
321 //
322 // Preconditions:
323 //
324 // Parameters: pmsg pointer to BMessage where the settings
325 // will be stored
326 //
327 // Postconditions:
328 //
329 // Returns: B_OK if there were no errors or an error code from
330 // BMessage::RemoveName() or BMessage::AddBool() if there were
331 // errors
332 // ---------------------------------------------------------------
333 status_t
GetConfigurationMessage(BMessage * pmsg)334 TranslatorSettings::GetConfigurationMessage(BMessage *pmsg)
335 {
336 status_t result = B_BAD_VALUE;
337
338 if (pmsg) {
339 int32 i;
340 for (i = 0; i < fDefCount; i++) {
341 result = pmsg->RemoveName(fDefaults[i].name);
342 if (result != B_OK && result != B_NAME_NOT_FOUND)
343 break;
344 }
345 if (i == fDefCount) {
346 fLock.Lock();
347 result = B_OK;
348
349 const TranSetting *defs = fDefaults;
350 for (i = 0; i < fDefCount && result >= B_OK; i++) {
351 switch (defs[i].dataType) {
352 case TRAN_SETTING_BOOL:
353 result = pmsg->AddBool(defs[i].name,
354 SetGetBool(defs[i].name));
355 break;
356
357 case TRAN_SETTING_INT32:
358 result = pmsg->AddInt32(defs[i].name,
359 SetGetInt32(defs[i].name));
360 break;
361
362 default:
363 // ASSERT here? Erase the bogus setting entry instead?
364 break;
365 }
366 }
367
368 fLock.Unlock();
369 }
370 }
371
372 return result;
373 }
374
375 // ---------------------------------------------------------------
376 // FindTranSetting
377 //
378 // Returns a pointer to the TranSetting with the given name
379 //
380 //
381 // Preconditions:
382 //
383 // Parameters: name name of the TranSetting to find
384 //
385 //
386 // Postconditions:
387 //
388 // Returns: NULL if the TranSetting cannot be found, or a pointer
389 // to the desired TranSetting if it is found
390 // ---------------------------------------------------------------
391 const TranSetting *
FindTranSetting(const char * name)392 TranslatorSettings::FindTranSetting(const char *name)
393 {
394 for (int32 i = 0; i < fDefCount; i++) {
395 if (!strcmp(fDefaults[i].name, name))
396 return fDefaults + i;
397 }
398 return NULL;
399 }
400
401 // ---------------------------------------------------------------
402 // SetGetBool
403 //
404 // Sets the state of the bool setting identified by the given name
405 //
406 //
407 // Preconditions:
408 //
409 // Parameters: name identifies the setting to set or get
410 //
411 // pbool the new value for the bool, or, if null,
412 // it indicates that the caller wants to Get
413 // rather than Set
414 //
415 // Postconditions:
416 //
417 // Returns: the prior value of the setting
418 // ---------------------------------------------------------------
419 bool
SetGetBool(const char * name,bool * pbool)420 TranslatorSettings::SetGetBool(const char *name, bool *pbool)
421 {
422 bool bprevValue;
423
424 fLock.Lock();
425
426 const TranSetting *def = FindTranSetting(name);
427 if (def) {
428 fSettingsMsg.FindBool(def->name, &bprevValue);
429 if (pbool)
430 fSettingsMsg.ReplaceBool(def->name, *pbool);
431 } else
432 bprevValue = false;
433
434 fLock.Unlock();
435
436 return bprevValue;
437 }
438
439 // ---------------------------------------------------------------
440 // SetGetInt32
441 //
442 // Sets the state of the int32 setting identified by the given name
443 //
444 //
445 // Preconditions:
446 //
447 // Parameters: name identifies the setting to set or get
448 //
449 // pint32 the new value for the setting, or, if null,
450 // it indicates that the caller wants to Get
451 // rather than Set
452 //
453 // Postconditions:
454 //
455 // Returns: the prior value of the setting
456 // ---------------------------------------------------------------
457 int32
SetGetInt32(const char * name,int32 * pint32)458 TranslatorSettings::SetGetInt32(const char *name, int32 *pint32)
459 {
460 int32 prevValue;
461
462 fLock.Lock();
463
464 const TranSetting *def = FindTranSetting(name);
465 if (def) {
466 fSettingsMsg.FindInt32(def->name, &prevValue);
467 if (pint32)
468 fSettingsMsg.ReplaceInt32(def->name, *pint32);
469 } else
470 prevValue = 0;
471
472 fLock.Unlock();
473
474 return prevValue;
475 }
476
477