xref: /haiku/src/kits/tracker/ViewState.cpp (revision 4bd0c1066b227cec4b79883bdef697c7a27f2e90)
1 /*
2 Open Tracker License
3 
4 Terms and Conditions
5 
6 Copyright (c) 1991-2000, Be Incorporated. All rights reserved.
7 
8 Permission is hereby granted, free of charge, to any person obtaining a copy of
9 this software and associated documentation files (the "Software"), to deal in
10 the Software without restriction, including without limitation the rights to
11 use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
12 of the Software, and to permit persons to whom the Software is furnished to do
13 so, subject to the following conditions:
14 
15 The above copyright notice and this permission notice applies to all licensees
16 and shall be included in all copies or substantial portions of the Software.
17 
18 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY,
20 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
21 BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
22 AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION
23 WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 
25 Except as contained in this notice, the name of Be Incorporated shall not be
26 used in advertising or otherwise to promote the sale, use or other dealings in
27 this Software without prior written authorization from Be Incorporated.
28 
29 Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered trademarks
30 of Be Incorporated in the United States and other countries. Other brand product
31 names are registered trademarks or trademarks of their respective holders.
32 All rights reserved.
33 */
34 
35 
36 #include <Debug.h>
37 #include <AppDefs.h>
38 #include <InterfaceDefs.h>
39 
40 #include "Attributes.h"
41 #include "Commands.h"
42 #include "PoseView.h"
43 #include "Utilities.h"
44 #include "ViewState.h"
45 
46 #include <new>
47 #include <string.h>
48 #include <stdlib.h>
49 
50 
51 const char* kColumnVersionName = "BColumn:version";
52 const char* kColumnTitleName = "BColumn:fTitle";
53 const char* kColumnOffsetName = "BColumn:fOffset";
54 const char* kColumnWidthName = "BColumn:fWidth";
55 const char* kColumnAlignmentName = "BColumn:fAlignment";
56 const char* kColumnAttrName = "BColumn:fAttrName";
57 const char* kColumnAttrHashName = "BColumn:fAttrHash";
58 const char* kColumnAttrTypeName = "BColumn:fAttrType";
59 const char* kColumnDisplayAsName = "BColumn:fDisplayAs";
60 const char* kColumnStatFieldName = "BColumn:fStatField";
61 const char* kColumnEditableName = "BColumn:fEditable";
62 
63 const char* kViewStateVersionName = "ViewState:version";
64 const char* kViewStateViewModeName = "ViewState:fViewMode";
65 const char* kViewStateLastIconModeName = "ViewState:fLastIconMode";
66 const char* kViewStateListOriginName = "ViewState:fListOrigin";
67 const char* kViewStateIconOriginName = "ViewState:fIconOrigin";
68 const char* kViewStatePrimarySortAttrName = "ViewState:fPrimarySortAttr";
69 const char* kViewStatePrimarySortTypeName = "ViewState:fPrimarySortType";
70 const char* kViewStateSecondarySortAttrName = "ViewState:fSecondarySortAttr";
71 const char* kViewStateSecondarySortTypeName = "ViewState:fSecondarySortType";
72 const char* kViewStateReverseSortName = "ViewState:fReverseSort";
73 const char* kViewStateIconSizeName = "ViewState:fIconSize";
74 const char* kViewStateLastIconSizeName = "ViewState:fLastIconSize";
75 
76 
77 static const int32 kColumnStateMinArchiveVersion = 21;
78 	// bump version when layout changes
79 
80 
81 //	#pragma mark - BColumn
82 
83 
84 BColumn::BColumn(const char* title, float offset, float width,
85 	alignment align, const char* attributeName, uint32 attrType,
86 	const char* displayAs, bool statField, bool editable)
87 {
88 	_Init(title, offset, width, align, attributeName, attrType, displayAs,
89 		statField, editable);
90 }
91 
92 
93 BColumn::BColumn(const char* title, float offset, float width,
94 	alignment align, const char* attributeName, uint32 attrType,
95 	bool statField, bool editable)
96 {
97 	_Init(title, offset, width, align, attributeName, attrType, NULL,
98 		statField, editable);
99 }
100 
101 
102 BColumn::~BColumn()
103 {
104 }
105 
106 
107 BColumn::BColumn(BMallocIO* stream, int32 version, bool endianSwap)
108 {
109 	StringFromStream(&fTitle, stream, endianSwap);
110 	stream->Read(&fOffset, sizeof(float));
111 	stream->Read(&fWidth, sizeof(float));
112 	stream->Read(&fAlignment, sizeof(alignment));
113 	StringFromStream(&fAttrName, stream, endianSwap);
114 	stream->Read(&fAttrHash, sizeof(uint32));
115 	stream->Read(&fAttrType, sizeof(uint32));
116 	stream->Read(&fStatField, sizeof(bool));
117 	stream->Read(&fEditable, sizeof(bool));
118 	if (version == kColumnStateArchiveVersion)
119 		StringFromStream(&fDisplayAs, stream, endianSwap);
120 
121 	if (endianSwap) {
122 		PRINT(("endian swapping column\n"));
123 		fOffset = B_SWAP_FLOAT(fOffset);
124 		fWidth = B_SWAP_FLOAT(fWidth);
125 		STATIC_ASSERT(sizeof(alignment) == sizeof(int32));
126 		fAlignment = (alignment)B_SWAP_INT32(fAlignment);
127 		fAttrHash = B_SWAP_INT32(fAttrHash);
128 		fAttrType = B_SWAP_INT32(fAttrType);
129 	}
130 }
131 
132 
133 BColumn::BColumn(const BMessage &message, int32 index)
134 {
135 	if (message.FindString(kColumnTitleName, index, &fTitle) != B_OK)
136 		fTitle.SetTo(B_EMPTY_STRING);
137 
138 	if (message.FindFloat(kColumnOffsetName, index, &fOffset) != B_OK)
139 		fOffset = -1.0f;
140 
141 	if (message.FindFloat(kColumnWidthName, index, &fWidth) != B_OK)
142 		fWidth = -1.0f;
143 
144 	if (message.FindInt32(kColumnAlignmentName, index, (int32*)&fAlignment)
145 			!= B_OK) {
146 		fAlignment = B_ALIGN_LEFT;
147 	}
148 
149 	if (message.FindString(kColumnAttrName, index, &fAttrName) != B_OK)
150 		fAttrName = BString(B_EMPTY_STRING);
151 
152 	if (message.FindInt32(kColumnAttrHashName, index, (int32*)&fAttrHash)
153 			!= B_OK) {
154 		fAttrHash = 0;
155 	}
156 
157 	if (message.FindInt32(kColumnAttrTypeName, index, (int32*)&fAttrType)
158 			!= B_OK) {
159 		fAttrType = 0;
160 	}
161 
162 	if (message.FindString(kColumnDisplayAsName, index, &fDisplayAs) != B_OK)
163 		fDisplayAs.SetTo(B_EMPTY_STRING);
164 
165 	if (message.FindBool(kColumnStatFieldName, index, &fStatField) != B_OK)
166 		fStatField = false;
167 
168 	if (message.FindBool(kColumnEditableName, index, &fEditable) != B_OK)
169 		fEditable = false;
170 
171 }
172 
173 
174 void
175 BColumn::_Init(const char* title, float offset, float width,
176 	alignment align, const char* attributeName, uint32 attrType,
177 	const char* displayAs, bool statField, bool editable)
178 {
179 	fTitle = title;
180 	fAttrName = attributeName;
181 	fDisplayAs = displayAs;
182 	fOffset = offset;
183 	fWidth = width;
184 	fAlignment = align;
185 	fAttrHash = AttrHashString(attributeName, attrType);
186 	fAttrType = attrType;
187 	fStatField = statField;
188 	fEditable = editable;
189 }
190 
191 
192 BColumn*
193 BColumn::InstantiateFromStream(BMallocIO* stream, bool endianSwap)
194 {
195 	// compare stream header in canonical form
196 
197 	// we can't use ValidateStream(), as we preserve backwards compatibility
198 	int32 version;
199 	uint32 key;
200 	if (stream->Read(&key, sizeof(uint32)) <= 0
201 		|| stream->Read(&version, sizeof(int32)) <=0)
202 		return 0;
203 
204 	if (endianSwap) {
205 		key = SwapUInt32(key);
206 		version = SwapInt32(version);
207 	}
208 
209 	if (key != AttrHashString("BColumn", B_OBJECT_TYPE)
210 		|| version < kColumnStateMinArchiveVersion)
211 		return 0;
212 
213 //	PRINT(("instantiating column, %s\n", endianSwap ? "endian swapping," : ""));
214 	return _Sanitize(new (std::nothrow) BColumn(stream, version, endianSwap));
215 }
216 
217 
218 BColumn*
219 BColumn::InstantiateFromMessage(const BMessage &message, int32 index)
220 {
221 	int32 version = kColumnStateArchiveVersion;
222 	int32 messageVersion;
223 
224 	if (message.FindInt32(kColumnVersionName, index, &messageVersion) != B_OK)
225 		return NULL;
226 
227 	if (version != messageVersion)
228 		return NULL;
229 
230 	return _Sanitize(new (std::nothrow) BColumn(message, index));
231 }
232 
233 
234 void
235 BColumn::ArchiveToStream(BMallocIO* stream) const
236 {
237 	// write class identifier and version info
238 	uint32 key = AttrHashString("BColumn", B_OBJECT_TYPE);
239 	stream->Write(&key, sizeof(uint32));
240 	int32 version = kColumnStateArchiveVersion;
241 	stream->Write(&version, sizeof(int32));
242 
243 //	PRINT(("ArchiveToStream column, key %x, version %d\n", key, version));
244 
245 	StringToStream(&fTitle, stream);
246 	stream->Write(&fOffset, sizeof(float));
247 	stream->Write(&fWidth, sizeof(float));
248 	stream->Write(&fAlignment, sizeof(alignment));
249 	StringToStream(&fAttrName, stream);
250 	stream->Write(&fAttrHash, sizeof(uint32));
251 	stream->Write(&fAttrType, sizeof(uint32));
252 	stream->Write(&fStatField, sizeof(bool));
253 	stream->Write(&fEditable, sizeof(bool));
254 	StringToStream(&fDisplayAs, stream);
255 }
256 
257 
258 void
259 BColumn::ArchiveToMessage(BMessage &message) const
260 {
261 	message.AddInt32(kColumnVersionName, kColumnStateArchiveVersion);
262 
263 	message.AddString(kColumnTitleName, fTitle);
264 	message.AddFloat(kColumnOffsetName, fOffset);
265 	message.AddFloat(kColumnWidthName, fWidth);
266 	message.AddInt32(kColumnAlignmentName, fAlignment);
267 	message.AddString(kColumnAttrName, fAttrName);
268 	message.AddInt32(kColumnAttrHashName, static_cast<int32>(fAttrHash));
269 	message.AddInt32(kColumnAttrTypeName, static_cast<int32>(fAttrType));
270 	message.AddString(kColumnDisplayAsName, fDisplayAs.String());
271 	message.AddBool(kColumnStatFieldName, fStatField);
272 	message.AddBool(kColumnEditableName, fEditable);
273 }
274 
275 
276 BColumn*
277 BColumn::_Sanitize(BColumn* column)
278 {
279 	if (column == NULL)
280 		return NULL;
281 
282 	// sanity-check the resulting column
283 	if (column->fTitle.Length() > 500
284 		|| column->fOffset < 0
285 		|| column->fOffset > 10000
286 		|| column->fWidth < 0
287 		|| column->fWidth > 10000
288 		|| (int32)column->fAlignment < B_ALIGN_LEFT
289 		|| (int32)column->fAlignment > B_ALIGN_CENTER
290 		|| column->fAttrName.Length() > 500) {
291 		PRINT(("column data not valid\n"));
292 		delete column;
293 		return NULL;
294 	}
295 #if DEBUG
296 // TODO: Whatever this is supposed to mean, fix it.
297 //	else if (endianSwap)
298 //		PRINT(("Instantiated foreign column ok\n"));
299 #endif
300 
301 	return column;
302 }
303 
304 
305 //	#pragma mark - BViewState
306 
307 
308 BViewState::BViewState()
309 {
310 	_Init();
311 	_StorePreviousState();
312 }
313 
314 
315 BViewState::BViewState(BMallocIO* stream, bool endianSwap)
316 {
317 	_Init();
318 	stream->Read(&fViewMode, sizeof(uint32));
319 	stream->Read(&fLastIconMode, sizeof(uint32));
320 	stream->Read(&fListOrigin, sizeof(BPoint));
321 	stream->Read(&fIconOrigin, sizeof(BPoint));
322 	stream->Read(&fPrimarySortAttr, sizeof(uint32));
323 	stream->Read(&fPrimarySortType, sizeof(uint32));
324 	stream->Read(&fSecondarySortAttr, sizeof(uint32));
325 	stream->Read(&fSecondarySortType, sizeof(uint32));
326 	stream->Read(&fReverseSort, sizeof(bool));
327 	stream->Read(&fIconSize, sizeof(uint32));
328 	stream->Read(&fLastIconSize, sizeof(uint32));
329 
330 	if (endianSwap) {
331 		PRINT(("endian swapping view state\n"));
332 		fViewMode = B_SWAP_INT32(fViewMode);
333 		fLastIconMode = B_SWAP_INT32(fLastIconMode);
334 		fIconSize = B_SWAP_INT32(fIconSize);
335 		fLastIconSize = B_SWAP_INT32(fLastIconSize);
336 		swap_data(B_POINT_TYPE, &fListOrigin,
337 			sizeof(fListOrigin), B_SWAP_ALWAYS);
338 		swap_data(B_POINT_TYPE, &fIconOrigin,
339 			sizeof(fIconOrigin), B_SWAP_ALWAYS);
340 		fPrimarySortAttr = B_SWAP_INT32(fPrimarySortAttr);
341 		fSecondarySortAttr = B_SWAP_INT32(fSecondarySortAttr);
342 		fPrimarySortType = B_SWAP_INT32(fPrimarySortType);
343 		fSecondarySortType = B_SWAP_INT32(fSecondarySortType);
344 	}
345 
346 	_StorePreviousState();
347 	_Sanitize(this, true);
348 }
349 
350 
351 BViewState::BViewState(const BMessage &message)
352 {
353 	_Init();
354 	message.FindInt32(kViewStateViewModeName, (int32*)&fViewMode);
355 	message.FindInt32(kViewStateLastIconModeName, (int32*)&fLastIconMode);
356 	message.FindInt32(kViewStateLastIconSizeName,(int32*)&fLastIconSize);
357 	message.FindInt32(kViewStateIconSizeName, (int32*)&fIconSize);
358 	message.FindPoint(kViewStateListOriginName, &fListOrigin);
359 	message.FindPoint(kViewStateIconOriginName, &fIconOrigin);
360 	message.FindInt32(kViewStatePrimarySortAttrName,
361 		(int32*)&fPrimarySortAttr);
362 	message.FindInt32(kViewStatePrimarySortTypeName,
363 		(int32*)&fPrimarySortType);
364 	message.FindInt32(kViewStateSecondarySortAttrName,
365 		(int32*)&fSecondarySortAttr);
366 	message.FindInt32(kViewStateSecondarySortTypeName,
367 		(int32*)&fSecondarySortType);
368 	message.FindBool(kViewStateReverseSortName, &fReverseSort);
369 
370 	_StorePreviousState();
371 	_Sanitize(this, true);
372 }
373 
374 
375 void
376 BViewState::ArchiveToStream(BMallocIO* stream) const
377 {
378 	// write class identifier and verison info
379 	uint32 key = AttrHashString("BViewState", B_OBJECT_TYPE);
380 	stream->Write(&key, sizeof(key));
381 	int32 version = kViewStateArchiveVersion;
382 	stream->Write(&version, sizeof(version));
383 
384 	stream->Write(&fViewMode, sizeof(uint32));
385 	stream->Write(&fLastIconMode, sizeof(uint32));
386 	stream->Write(&fListOrigin, sizeof(BPoint));
387 	stream->Write(&fIconOrigin, sizeof(BPoint));
388 	stream->Write(&fPrimarySortAttr, sizeof(uint32));
389 	stream->Write(&fPrimarySortType, sizeof(uint32));
390 	stream->Write(&fSecondarySortAttr, sizeof(uint32));
391 	stream->Write(&fSecondarySortType, sizeof(uint32));
392 	stream->Write(&fReverseSort, sizeof(bool));
393 	stream->Write(&fIconSize, sizeof(uint32));
394 	stream->Write(&fLastIconSize, sizeof(uint32));
395 }
396 
397 
398 void
399 BViewState::ArchiveToMessage(BMessage &message) const
400 {
401 	message.AddInt32(kViewStateVersionName, kViewStateArchiveVersion);
402 
403 	message.AddInt32(kViewStateViewModeName, static_cast<int32>(fViewMode));
404 	message.AddInt32(kViewStateLastIconModeName,
405 		static_cast<int32>(fLastIconMode));
406 	message.AddPoint(kViewStateListOriginName, fListOrigin);
407 	message.AddPoint(kViewStateIconOriginName, fIconOrigin);
408 	message.AddInt32(kViewStatePrimarySortAttrName,
409 		static_cast<int32>(fPrimarySortAttr));
410 	message.AddInt32(kViewStatePrimarySortTypeName,
411 		static_cast<int32>(fPrimarySortType));
412 	message.AddInt32(kViewStateSecondarySortAttrName,
413 		static_cast<int32>(fSecondarySortAttr));
414 	message.AddInt32(kViewStateSecondarySortTypeName,
415 		static_cast<int32>(fSecondarySortType));
416 	message.AddBool(kViewStateReverseSortName, fReverseSort);
417 	message.AddInt32(kViewStateIconSizeName, static_cast<int32>(fIconSize));
418 	message.AddInt32(kViewStateLastIconSizeName,
419 		static_cast<int32>(fLastIconSize));
420 }
421 
422 
423 BViewState*
424 BViewState::InstantiateFromStream(BMallocIO* stream, bool endianSwap)
425 {
426 	// compare stream header in canonical form
427 	uint32 key = AttrHashString("BViewState", B_OBJECT_TYPE);
428 	int32 version = kViewStateArchiveVersion;
429 
430 	if (endianSwap) {
431 		key = SwapUInt32(key);
432 		version = SwapInt32(version);
433 	}
434 
435 	if (!ValidateStream(stream, key, version))
436 		return NULL;
437 
438 	return _Sanitize(new (std::nothrow) BViewState(stream, endianSwap));
439 }
440 
441 
442 BViewState*
443 BViewState::InstantiateFromMessage(const BMessage &message)
444 {
445 	int32 version = kViewStateArchiveVersion;
446 
447 	int32 messageVersion;
448 	if (message.FindInt32(kViewStateVersionName, &messageVersion) != B_OK)
449 		return NULL;
450 
451 	if (version != messageVersion)
452 		return NULL;
453 
454 	return _Sanitize(new (std::nothrow) BViewState(message));
455 }
456 
457 
458 void
459 BViewState::_Init()
460 {
461 	fViewMode = kListMode;
462 	fLastIconMode = 0;
463 	fIconSize = B_MINI_ICON;
464 	fLastIconSize = B_MINI_ICON;
465 	fListOrigin.Set(0, 0);
466 	fIconOrigin.Set(0, 0);
467 	fPrimarySortAttr = AttrHashString(kAttrStatName, B_STRING_TYPE);
468 	fPrimarySortType = B_STRING_TYPE;
469 	fSecondarySortAttr = 0;
470 	fSecondarySortType = 0;
471 	fReverseSort = false;
472 }
473 
474 
475 void
476 BViewState::_StorePreviousState()
477 {
478 	fPreviousViewMode = fViewMode;
479 	fPreviousLastIconMode = fLastIconMode;
480 	fPreviousIconSize = fIconSize;
481 	fPreviousLastIconSize = fLastIconSize;
482 	fPreviousListOrigin = fListOrigin;
483 	fPreviousIconOrigin = fIconOrigin;
484 	fPreviousPrimarySortAttr = fPrimarySortAttr;
485 	fPreviousSecondarySortAttr = fSecondarySortAttr;
486 	fPreviousPrimarySortType = fPrimarySortType;
487 	fPreviousSecondarySortType = fSecondarySortType;
488 	fPreviousReverseSort = fReverseSort;
489 }
490 
491 
492 BViewState*
493 BViewState::_Sanitize(BViewState* state, bool fixOnly)
494 {
495 	if (state == NULL)
496 		return NULL;
497 
498 	if (state->fViewMode == kListMode) {
499 		if (state->fListOrigin.x < 0)
500 			state->fListOrigin.x = 0;
501 
502 		if (state->fListOrigin.y < 0)
503 			state->fListOrigin.y = 0;
504 	}
505 	if (state->fIconSize < 16)
506 		state->fIconSize = 16;
507 
508 	if (state->fIconSize > 128)
509 		state->fIconSize = 128;
510 
511 	if (state->fLastIconSize < 16)
512 		state->fLastIconSize = 16;
513 
514 	if (state->fLastIconSize > 128)
515 		state->fLastIconSize = 128;
516 
517 	if (fixOnly)
518 		return state;
519 
520 	// do a sanity check here
521 	if ((state->fViewMode != kListMode
522 			&& state->fViewMode != kIconMode
523 			&& state->fViewMode != kMiniIconMode
524 			&& state->fViewMode != 0)
525 		|| (state->fLastIconMode != kListMode
526 			&& state->fLastIconMode != kIconMode
527 			&& state->fLastIconMode != kMiniIconMode
528 			&& state->fLastIconMode != 0)) {
529 		PRINT(("Bad data instantiating ViewState, view mode %" B_PRIx32
530 			", lastIconMode %" B_PRIx32 "\n", state->fViewMode,
531 			state->fLastIconMode));
532 
533 		delete state;
534 		return NULL;
535 	}
536 #if DEBUG
537 // TODO: Whatever this is supposed to mean, fix it.
538 //	else if (endianSwap)
539 //		PRINT(("Instantiated foreign view state ok\n"));
540 #endif
541 
542 	return state;
543 }
544