]> git.saurik.com Git - wxWidgets.git/blob - src/osx/cocoa/dataview.mm
Use virtual functions to convert NSObject to the correct type in wxDVC.
[wxWidgets.git] / src / osx / cocoa / dataview.mm
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/osx/cocoa/dataview.mm
3 // Purpose: wxDataView
4 // Author:
5 // Modified by:
6 // Created: 2009-01-31
7 // RCS-ID: $Id: dataview.mm$
8 // Copyright:
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
11
12 #include "wx/wxprec.h"
13
14 #if (wxUSE_DATAVIEWCTRL == 1) && !defined(wxUSE_GENERICDATAVIEWCTRL)
15
16 #ifndef WX_PRECOMP
17 #include "wx/app.h"
18 #include "wx/toplevel.h"
19 #include "wx/font.h"
20 #include "wx/settings.h"
21 #include "wx/utils.h"
22 #endif
23
24 #include "wx/osx/cocoa/dataview.h"
25 #include "wx/osx/private.h"
26 #include "wx/renderer.h"
27
28 // ============================================================================
29 // Constants used locally
30 // ============================================================================
31
32 #define DataViewPboardType @"OutlineViewItem"
33
34 // ============================================================================
35 // Classes used locally in dataview.mm
36 // ============================================================================
37 @interface wxCustomRendererObject : NSObject <NSCopying>
38 {
39 @public
40 wxDataViewCustomRenderer* customRenderer; // not owned by the class
41 }
42
43 -(id) init;
44 -(id) initWithRenderer:(wxDataViewCustomRenderer*)renderer;
45 @end
46
47 @implementation wxCustomRendererObject
48
49 -(id) init
50 {
51 self = [super init];
52 if (self != nil)
53 {
54 customRenderer = NULL;
55 }
56 return self;
57 }
58
59 -(id) initWithRenderer:(wxDataViewCustomRenderer*)renderer
60 {
61 self = [super init];
62 if (self != nil)
63 {
64 customRenderer = renderer;
65 }
66 return self;
67 }
68
69 -(id) copyWithZone:(NSZone*)zone
70 {
71 wxCustomRendererObject* copy;
72
73 copy = [[[self class] allocWithZone:zone] init];
74 copy->customRenderer = customRenderer;
75
76 return copy;
77 }
78 @end
79
80 // ============================================================================
81 // local helpers
82 // ============================================================================
83
84 namespace
85 {
86
87 // convert from NSObject to different C++ types: all these functions check
88 // that the conversion really makes sense and assert if it doesn't
89 wxString ObjectToString(NSObject *object)
90 {
91 wxCHECK_MSG( [object isKindOfClass:[NSString class]], "",
92 wxString::Format
93 (
94 "string expected but got %s",
95 wxCFStringRef::AsString([object className])
96 ));
97
98 return wxCFStringRef([((NSString*) object) retain]).AsString();
99 }
100
101 bool ObjectToBool(NSObject *object)
102 {
103 // actually the value must be of NSCFBoolean class but it's private so we
104 // can't check for it directly
105 wxCHECK_MSG( [object isKindOfClass:[NSNumber class]], false,
106 wxString::Format
107 (
108 "number expected but got %s",
109 wxCFStringRef::AsString([object className])
110 ));
111
112 return [(NSNumber *)object boolValue];
113 }
114
115 long ObjectToLong(NSObject *object)
116 {
117 wxCHECK_MSG( [object isKindOfClass:[NSNumber class]], -1,
118 wxString::Format
119 (
120 "number expected but got %s",
121 wxCFStringRef::AsString([object className])
122 ));
123
124 return [(NSNumber *)object longValue];
125 }
126
127 wxDateTime ObjectToDate(NSObject *object)
128 {
129 wxCHECK_MSG( [object isKindOfClass:[NSDate class]], wxInvalidDateTime,
130 wxString::Format
131 (
132 "date expected but got %s",
133 wxCFStringRef::AsString([object className])
134 ));
135
136 // get the number of seconds since 1970-01-01 UTC and this is the only
137 // way to convert a double to a wxLongLong
138 const wxLongLong seconds = [((NSDate*) object) timeIntervalSince1970];
139
140 wxDateTime dt(1, wxDateTime::Jan, 1970);
141 dt.Add(wxTimeSpan(0,0,seconds));
142
143 // the user has entered a date in the local timezone but seconds
144 // contains the number of seconds from date in the local timezone
145 // since 1970-01-01 UTC; therefore, the timezone information has to be
146 // transferred to wxWidgets, too:
147 dt.MakeFromTimezone(wxDateTime::UTC);
148
149 return dt;
150 }
151
152 NSInteger CompareItems(id item1, id item2, void* context)
153 {
154 NSArray* const sortDescriptors = (NSArray*) context;
155
156 NSUInteger const count = [sortDescriptors count];
157
158 NSInteger result = NSOrderedSame;
159 for ( NSUInteger i = 0; i < count && result == NSOrderedSame; ++i )
160 {
161 // constant definition for abbreviational purposes:
162 wxSortDescriptorObject* const
163 sortDescriptor = (wxSortDescriptorObject*)
164 [sortDescriptors objectAtIndex:i];
165
166 int rc = [sortDescriptor modelPtr]->Compare
167 (
168 wxDataViewItem([((wxPointerObject*) item1) pointer]),
169 wxDataViewItem([((wxPointerObject*) item2) pointer]),
170 [sortDescriptor columnPtr]->GetModelColumn(),
171 [sortDescriptor ascending] == YES
172 );
173
174 if ( rc < 0 )
175 result = NSOrderedAscending;
176 else if ( rc > 0 )
177 result = NSOrderedDescending;
178 }
179
180 return result;
181 }
182
183 NSTextAlignment ConvertToNativeHorizontalTextAlignment(int alignment)
184 {
185 if (alignment & wxALIGN_CENTER_HORIZONTAL)
186 return NSCenterTextAlignment;
187 else if (alignment & wxALIGN_RIGHT)
188 return NSRightTextAlignment;
189 else
190 return NSLeftTextAlignment;
191 }
192
193 NSTableColumn* CreateNativeColumn(const wxDataViewColumn *column)
194 {
195 wxDataViewRenderer * const renderer = column->GetRenderer();
196
197 wxCHECK_MSG( renderer, NULL, "column should have a renderer" );
198
199 NSTableColumn * const nativeColumn(
200 [[NSTableColumn alloc] initWithIdentifier:
201 [[[wxPointerObject alloc] initWithPointer:
202 const_cast<wxDataViewColumn*>(column)]
203 autorelease]]
204 );
205
206 // setting the size related parameters:
207 if (column->IsResizeable())
208 {
209 [nativeColumn setResizingMask:NSTableColumnUserResizingMask];
210 [nativeColumn setMinWidth:column->GetMinWidth()];
211 [nativeColumn setMaxWidth:column->GetMaxWidth()];
212 }
213 else
214 {
215 [nativeColumn setResizingMask:NSTableColumnNoResizing];
216 [nativeColumn setMinWidth:column->GetWidth()];
217 [nativeColumn setMaxWidth:column->GetWidth()];
218 }
219 [nativeColumn setWidth:column->GetWidth()];
220
221 #if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_5
222 // setting the visibility:
223 [nativeColumn setHidden:static_cast<BOOL>(column->IsHidden())];
224 #endif
225
226 wxDataViewRendererNativeData * const renderData = renderer->GetNativeData();
227
228 // setting the header:
229 [[nativeColumn headerCell] setAlignment:
230 ConvertToNativeHorizontalTextAlignment(column->GetAlignment())];
231 [[nativeColumn headerCell] setStringValue:
232 [[wxCFStringRef(column->GetTitle()).AsNSString() retain] autorelease]];
233 renderData->ApplyLineBreakMode([nativeColumn headerCell]);
234
235 // setting data cell's properties:
236 [[nativeColumn dataCell] setWraps:NO];
237 // setting the default data cell:
238 [nativeColumn setDataCell:renderData->GetColumnCell()];
239 // setting the editablility:
240 const bool isEditable = renderer->GetMode() == wxDATAVIEW_CELL_EDITABLE;
241
242 [nativeColumn setEditable:isEditable];
243 [[nativeColumn dataCell] setEditable:isEditable];
244
245 return nativeColumn;
246 }
247
248 } // anonymous namespace
249
250 // ============================================================================
251 // Public helper functions for dataview implementation on OSX
252 // ============================================================================
253
254 wxWidgetImplType* CreateDataView(wxWindowMac* wxpeer,
255 wxWindowMac* WXUNUSED(parent),
256 wxWindowID WXUNUSED(id),
257 const wxPoint& pos,
258 const wxSize& size,
259 long style,
260 long WXUNUSED(extraStyle))
261 {
262 return new wxCocoaDataViewControl(wxpeer,pos,size,style);
263 }
264
265 // ============================================================================
266 // wxPointerObject
267 // ============================================================================
268
269 @implementation wxPointerObject
270
271 -(id) init
272 {
273 self = [super init];
274 if (self != nil)
275 self->pointer = NULL;
276 return self;
277 }
278
279 -(id) initWithPointer:(void*) initPointer
280 {
281 self = [super init];
282 if (self != nil)
283 self->pointer = initPointer;
284 return self;
285 }
286
287 //
288 // inherited methods from NSObject
289 //
290 -(BOOL) isEqual:(id)object
291 {
292 return (object != nil) &&
293 ([object isKindOfClass:[wxPointerObject class]]) &&
294 (pointer == [((wxPointerObject*) object) pointer]);
295 }
296
297 -(NSUInteger) hash
298 {
299 return (NSUInteger) pointer;
300 }
301
302 -(void*) pointer
303 {
304 return pointer;
305 }
306
307 -(void) setPointer:(void*) newPointer
308 {
309 pointer = newPointer;
310 }
311
312 @end
313
314 // ============================================================================
315 // wxSortDescriptorObject
316 // ============================================================================
317
318 @implementation wxSortDescriptorObject
319 -(id) init
320 {
321 self = [super init];
322 if (self != nil)
323 {
324 columnPtr = NULL;
325 modelPtr = NULL;
326 }
327 return self;
328 }
329
330 -(id)
331 initWithModelPtr:(wxDataViewModel*)initModelPtr
332 sortingColumnPtr:(wxDataViewColumn*)initColumnPtr
333 ascending:(BOOL)sortAscending
334 {
335 self = [super initWithKey:@"dummy" ascending:sortAscending];
336 if (self != nil)
337 {
338 columnPtr = initColumnPtr;
339 modelPtr = initModelPtr;
340 }
341 return self;
342 }
343
344 -(id) copyWithZone:(NSZone*)zone
345 {
346 wxSortDescriptorObject* copy;
347
348
349 copy = [super copyWithZone:zone];
350 copy->columnPtr = columnPtr;
351 copy->modelPtr = modelPtr;
352
353 return copy;
354 }
355
356 //
357 // access to model column's index
358 //
359 -(wxDataViewColumn*) columnPtr
360 {
361 return columnPtr;
362 }
363
364 -(wxDataViewModel*) modelPtr
365 {
366 return modelPtr;
367 }
368
369 -(void) setColumnPtr:(wxDataViewColumn*)newColumnPtr
370 {
371 columnPtr = newColumnPtr;
372 }
373
374 -(void) setModelPtr:(wxDataViewModel*)newModelPtr
375 {
376 modelPtr = newModelPtr;
377 }
378
379 @end
380
381 // ============================================================================
382 // wxCocoaOutlineDataSource
383 // ============================================================================
384 @implementation wxCocoaOutlineDataSource
385
386 //
387 // constructors / destructor
388 //
389 -(id) init
390 {
391 self = [super init];
392 if (self != nil)
393 {
394 implementation = NULL;
395 model = NULL;
396
397 currentParentItem = nil;
398
399 children = [[NSMutableArray alloc] init];
400 items = [[NSMutableSet alloc] init];
401 }
402 return self;
403 }
404
405 -(void) dealloc
406 {
407 [currentParentItem release];
408
409 [children release];
410 [items release];
411
412 [super dealloc];
413 }
414
415 //
416 // methods of informal protocol:
417 //
418 -(BOOL)
419 outlineView:(NSOutlineView*)outlineView
420 acceptDrop:(id<NSDraggingInfo>)info
421 item:(id)item childIndex:(NSInteger)index
422 {
423 NSArray* supportedTypes(
424 [NSArray arrayWithObjects:DataViewPboardType,NSStringPboardType,nil]
425 );
426
427 NSPasteboard* pasteboard([info draggingPasteboard]);
428
429 NSString* bestType([pasteboard availableTypeFromArray:supportedTypes]);
430
431 if ( bestType == nil )
432 return FALSE;
433
434 wxDataViewCtrl * const dvc(implementation->GetDataViewCtrl());
435
436 wxCHECK_MSG( dvc, false,
437 "Pointer to data view control not set correctly." );
438 wxCHECK_MSG( dvc->GetModel(), false,
439 "Pointer to model not set correctly." );
440
441 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_DROP, dvc->GetId());
442 event.SetEventObject(dvc);
443 event.SetItem(wxDataViewItem([((wxPointerObject*) item) pointer]));
444 event.SetModel(dvc->GetModel());
445
446 BOOL dragSuccessful;
447 if ( [bestType compare:DataViewPboardType] == NSOrderedSame )
448 {
449 NSArray* dataArray((NSArray*)[pasteboard propertyListForType:DataViewPboardType]);
450 NSUInteger indexDraggedItem, noOfDraggedItems([dataArray count]);
451
452 indexDraggedItem = 0;
453 while (indexDraggedItem < noOfDraggedItems)
454 {
455 wxDataObjectComposite* dataObjects(implementation->GetDnDDataObjects((NSData*)[dataArray objectAtIndex:indexDraggedItem]));
456
457 if (dataObjects && (dataObjects->GetFormatCount() > 0))
458 {
459 wxMemoryBuffer buffer;
460
461 // copy data into data object:
462 event.SetDataObject(dataObjects);
463 event.SetDataFormat(implementation->GetDnDDataFormat(dataObjects));
464 // copy data into buffer:
465 dataObjects->GetDataHere(event.GetDataFormat().GetType(),buffer.GetWriteBuf(event.GetDataSize()));
466 buffer.UngetWriteBuf(event.GetDataSize());
467 event.SetDataBuffer(buffer.GetData());
468 // finally, send event:
469 if (dvc->HandleWindowEvent(event) && event.IsAllowed())
470 {
471 dragSuccessful = true;
472 ++indexDraggedItem;
473 }
474 else
475 {
476 dragSuccessful = true;
477 indexDraggedItem = noOfDraggedItems; // stop loop
478 }
479 }
480 else
481 {
482 dragSuccessful = false;
483 indexDraggedItem = noOfDraggedItems; // stop loop
484 }
485 // clean-up:
486 delete dataObjects;
487 }
488 }
489 else
490 {
491 CFDataRef osxData; // needed to convert internally used UTF-16 representation to a UTF-8 representation
492 wxDataObjectComposite* dataObjects (new wxDataObjectComposite());
493 wxTextDataObject* textDataObject(new wxTextDataObject());
494
495 osxData = ::CFStringCreateExternalRepresentation(kCFAllocatorDefault,(CFStringRef)[pasteboard stringForType:NSStringPboardType],kCFStringEncodingUTF8,32);
496 if (textDataObject->SetData(::CFDataGetLength(osxData),::CFDataGetBytePtr(osxData)))
497 dataObjects->Add(textDataObject);
498 else
499 delete textDataObject;
500 // send event if data could be copied:
501 if (dataObjects->GetFormatCount() > 0)
502 {
503 event.SetDataObject(dataObjects);
504 event.SetDataFormat(implementation->GetDnDDataFormat(dataObjects));
505 if (dvc->HandleWindowEvent(event) && event.IsAllowed())
506 dragSuccessful = true;
507 else
508 dragSuccessful = false;
509 }
510 else
511 dragSuccessful = false;
512 // clean up:
513 ::CFRelease(osxData);
514 delete dataObjects;
515 }
516 }
517
518 -(id) outlineView:(NSOutlineView*)outlineView child:(NSInteger)index ofItem:(id)item
519 {
520 if ((item == currentParentItem) && (index < ((NSInteger) [self getChildCount])))
521 return [self getChild:index];
522 else
523 {
524 wxDataViewItemArray dataViewChildren;
525
526 wxCHECK_MSG( model, 0, "Valid model in data source does not exist." );
527 (void) model->GetChildren((item == nil) ? wxDataViewItem() : wxDataViewItem([((wxPointerObject*) item) pointer]),dataViewChildren);
528 [self bufferItem:item withChildren:&dataViewChildren];
529 if ([sortDescriptors count] > 0)
530 [children sortUsingFunction:CompareItems context:sortDescriptors];
531 return [self getChild:index];
532 }
533 }
534
535 -(BOOL) outlineView:(NSOutlineView*)outlineView isItemExpandable:(id)item
536 {
537 wxCHECK_MSG( model, 0, "Valid model in data source does not exist." );
538 return model->IsContainer(wxDataViewItem([((wxPointerObject*) item) pointer]));
539 }
540
541 -(NSInteger) outlineView:(NSOutlineView*)outlineView numberOfChildrenOfItem:(id)item
542 {
543 NSInteger noOfChildren;
544
545 wxDataViewItemArray dataViewChildren;
546
547
548 wxCHECK_MSG( model, 0, "Valid model in data source does not exist." );
549 noOfChildren = model->GetChildren((item == nil) ? wxDataViewItem() : wxDataViewItem([((wxPointerObject*) item) pointer]),dataViewChildren);
550 [self bufferItem:item withChildren:&dataViewChildren];
551 if ([sortDescriptors count] > 0)
552 [children sortUsingFunction:CompareItems context:sortDescriptors];
553 return noOfChildren;
554 }
555
556 -(id)
557 outlineView:(NSOutlineView*)outlineView
558 objectValueForTableColumn:(NSTableColumn*)tableColumn
559 byItem:(id)item
560 {
561 wxDataViewColumn* col(static_cast<wxDataViewColumn*>([[tableColumn identifier] pointer]));
562
563 wxDataViewItem dataViewItem([((wxPointerObject*) item) pointer]);
564
565 wxVariant value;
566
567
568 wxCHECK_MSG( model, 0, "Valid model in data source does not exist." );
569 model->GetValue(value,dataViewItem,col->GetModelColumn());
570 col->GetRenderer()->SetValue(value);
571 return nil;
572 }
573
574 -(void)
575 outlineView:(NSOutlineView*)outlineView
576 setObjectValue:(id)object
577 forTableColumn:(NSTableColumn*)tableColumn
578 byItem:(id)item
579 {
580 wxDataViewColumn* col(static_cast<wxDataViewColumn*>([[tableColumn identifier] pointer]));
581
582 wxDataViewItem dataViewItem([((wxPointerObject*) item) pointer]);
583
584 col->GetRenderer()->
585 OSXOnCellChanged(object, dataViewItem, col->GetModelColumn());
586 }
587
588 -(void) outlineView:(NSOutlineView*)outlineView sortDescriptorsDidChange:(NSArray*)oldDescriptors
589 // Warning: the new sort descriptors are guaranteed to be only of type NSSortDescriptor! Therefore, the
590 // sort descriptors for the data source have to be converted.
591 {
592 NSArray* newDescriptors;
593
594 NSMutableArray* wxSortDescriptors;
595
596 NSUInteger noOfDescriptors;
597
598 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
599
600
601 // convert NSSortDescriptors to wxSortDescriptorObjects:
602 newDescriptors = [outlineView sortDescriptors];
603 noOfDescriptors = [newDescriptors count];
604 wxSortDescriptors = [NSMutableArray arrayWithCapacity:noOfDescriptors];
605 for (NSUInteger i=0; i<noOfDescriptors; ++i)
606 {
607 // constant definition for abbreviational purposes:
608 NSSortDescriptor* const newDescriptor = [newDescriptors objectAtIndex:i];
609
610 [wxSortDescriptors addObject:[[[wxSortDescriptorObject alloc] initWithModelPtr:model
611 sortingColumnPtr:dvc->GetColumn([[newDescriptor key] intValue])
612 ascending:[newDescriptor ascending]] autorelease]];
613 }
614 [[outlineView dataSource] setSortDescriptors:wxSortDescriptors];
615
616 // send first the event to wxWidgets that the sorting has changed so that the program can do special actions before
617 // the sorting actually starts:
618 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_COLUMN_SORTED,dvc->GetId()); // variable defintion
619
620 event.SetEventObject(dvc);
621 if (noOfDescriptors > 0)
622 {
623 // constant definition for abbreviational purposes:
624 wxDataViewColumn* const col = [[wxSortDescriptors objectAtIndex:0] columnPtr];
625
626 event.SetColumn(dvc->GetColumnPosition(col));
627 event.SetDataViewColumn(col);
628 }
629 dvc->GetEventHandler()->ProcessEvent(event);
630
631 // start re-ordering the data;
632 // children's buffer must be cleared first because it contains the old order:
633 [self clearChildren];
634 // sorting is done while reloading the data:
635 [outlineView reloadData];
636 }
637
638 -(NSDragOperation) outlineView:(NSOutlineView*)outlineView validateDrop:(id<NSDraggingInfo>)info proposedItem:(id)item proposedChildIndex:(NSInteger)index
639 {
640 NSArray* supportedTypes([NSArray arrayWithObjects:DataViewPboardType,NSStringPboardType,nil]);
641
642 NSPasteboard* pasteboard([info draggingPasteboard]);
643
644 NSString* bestType([pasteboard availableTypeFromArray:supportedTypes]);
645 if (bestType == nil)
646 return NSDragOperationNone;
647
648 NSDragOperation dragOperation;
649 wxDataViewCtrl* const dvc(implementation->GetDataViewCtrl());
650
651 wxCHECK_MSG(dvc, false, "Pointer to data view control not set correctly.");
652 wxCHECK_MSG(dvc->GetModel(), false, "Pointer to model not set correctly.");
653
654 wxDataViewEvent
655 event(wxEVT_COMMAND_DATAVIEW_ITEM_DROP_POSSIBLE,dvc->GetId());
656
657 event.SetEventObject(dvc);
658 event.SetItem(wxDataViewItem([((wxPointerObject*) item) pointer]));
659 event.SetModel(dvc->GetModel());
660 if ([bestType compare:DataViewPboardType] == NSOrderedSame)
661 {
662 NSArray* dataArray((NSArray*)[pasteboard propertyListForType:DataViewPboardType]);
663 NSUInteger indexDraggedItem, noOfDraggedItems([dataArray count]);
664
665 indexDraggedItem = 0;
666 while (indexDraggedItem < noOfDraggedItems)
667 {
668 wxDataObjectComposite* dataObjects(implementation->GetDnDDataObjects((NSData*)[dataArray objectAtIndex:indexDraggedItem]));
669
670 if (dataObjects && (dataObjects->GetFormatCount() > 0))
671 {
672 wxMemoryBuffer buffer;
673
674 // copy data into data object:
675 event.SetDataObject(dataObjects);
676 event.SetDataFormat(implementation->GetDnDDataFormat(dataObjects));
677 // copy data into buffer:
678 dataObjects->GetDataHere(event.GetDataFormat().GetType(),buffer.GetWriteBuf(event.GetDataSize()));
679 buffer.UngetWriteBuf(event.GetDataSize());
680 event.SetDataBuffer(buffer.GetData());
681 // finally, send event:
682 if (dvc->HandleWindowEvent(event) && event.IsAllowed())
683 {
684 dragOperation = NSDragOperationEvery;
685 ++indexDraggedItem;
686 }
687 else
688 {
689 dragOperation = NSDragOperationNone;
690 indexDraggedItem = noOfDraggedItems; // stop loop
691 }
692 }
693 else
694 {
695 dragOperation = NSDragOperationNone;
696 indexDraggedItem = noOfDraggedItems; // stop loop
697 }
698 // clean-up:
699 delete dataObjects;
700 }
701 }
702 else
703 {
704 CFDataRef osxData; // needed to convert internally used UTF-16 representation to a UTF-8 representation
705 wxDataObjectComposite* dataObjects (new wxDataObjectComposite());
706 wxTextDataObject* textDataObject(new wxTextDataObject());
707
708 osxData = ::CFStringCreateExternalRepresentation(kCFAllocatorDefault,(CFStringRef)[pasteboard stringForType:NSStringPboardType],kCFStringEncodingUTF8,32);
709 if (textDataObject->SetData(::CFDataGetLength(osxData),::CFDataGetBytePtr(osxData)))
710 dataObjects->Add(textDataObject);
711 else
712 delete textDataObject;
713 // send event if data could be copied:
714 if (dataObjects->GetFormatCount() > 0)
715 {
716 event.SetDataObject(dataObjects);
717 event.SetDataFormat(implementation->GetDnDDataFormat(dataObjects));
718 if (dvc->HandleWindowEvent(event) && event.IsAllowed())
719 dragOperation = NSDragOperationEvery;
720 else
721 dragOperation = NSDragOperationNone;
722 }
723 else
724 dragOperation = NSDragOperationNone;
725 // clean up:
726 ::CFRelease(osxData);
727 delete dataObjects;
728 }
729
730 return dragOperation;
731 }
732
733 -(BOOL) outlineView:(NSOutlineView*)outlineView writeItems:(NSArray*)writeItems toPasteboard:(NSPasteboard*)pasteboard
734 // the pasteboard will be filled up with an array containing the data as returned by the events (including the data type)
735 // and a concatenation of text (string) data; the text data will only be put onto the pasteboard if for all items a
736 // string representation exists
737 {
738 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
739
740 wxDataViewItemArray dataViewItems;
741
742
743 wxCHECK_MSG(dvc, false,"Pointer to data view control not set correctly.");
744 wxCHECK_MSG(dvc->GetModel(),false,"Pointer to model not set correctly.");
745
746 if ([writeItems count] > 0)
747 {
748 bool dataStringAvailable(true); // a flag indicating if for all items a data string is available
749 NSMutableArray* dataArray = [[NSMutableArray arrayWithCapacity:[writeItems count]] retain]; // data of all items
750 wxString dataString; // contains the string data of all items
751
752 // send a begin drag event for all selected items and proceed with dragging unless the event is vetoed:
753 wxDataViewEvent
754 event(wxEVT_COMMAND_DATAVIEW_ITEM_BEGIN_DRAG,dvc->GetId());
755
756 event.SetEventObject(dvc);
757 event.SetModel(dvc->GetModel());
758 for (size_t itemCounter=0; itemCounter<[writeItems count]; ++itemCounter)
759 {
760 bool itemStringAvailable(false); // a flag indicating if for the current item a string is available
761 wxDataObjectComposite* itemObject(new wxDataObjectComposite()); // data object for current item
762 wxString itemString; // contains the TAB concatenated data of an item
763
764 event.SetItem(wxDataViewItem([((wxPointerObject*) [writeItems objectAtIndex:itemCounter]) pointer]));
765 itemString = ::ConcatenateDataViewItemValues(dvc,event.GetItem());
766 itemObject->Add(new wxTextDataObject(itemString));
767 event.SetDataObject(itemObject);
768 // check if event has not been vetoed:
769 if (dvc->HandleWindowEvent(event) && event.IsAllowed() && (event.GetDataObject()->GetFormatCount() > 0))
770 {
771 // constant definition for abbreviational purposes:
772 size_t const noOfFormats = event.GetDataObject()->GetFormatCount();
773 // variable definition and initialization:
774 wxDataFormat* dataFormats(new wxDataFormat[noOfFormats]);
775
776 event.GetDataObject()->GetAllFormats(dataFormats,wxDataObject::Get);
777 for (size_t formatCounter=0; formatCounter<noOfFormats; ++formatCounter)
778 {
779 // constant definitions for abbreviational purposes:
780 wxDataFormatId const idDataFormat = dataFormats[formatCounter].GetType();
781 size_t const dataSize = event.GetDataObject()->GetDataSize(idDataFormat);
782 size_t const dataBufferSize = sizeof(wxDataFormatId)+dataSize;
783 // variable definitions (used in all case statements):
784 wxMemoryBuffer dataBuffer(dataBufferSize);
785
786 dataBuffer.AppendData(&idDataFormat,sizeof(wxDataFormatId));
787 switch (idDataFormat)
788 {
789 case wxDF_TEXT:
790 if (!itemStringAvailable) // otherwise wxDF_UNICODETEXT already filled up the string; and the UNICODE representation has priority
791 {
792 event.GetDataObject()->GetDataHere(wxDF_TEXT,dataBuffer.GetAppendBuf(dataSize));
793 dataBuffer.UngetAppendBuf(dataSize);
794 [dataArray addObject:[NSData dataWithBytes:dataBuffer.GetData() length:dataBufferSize]];
795 itemString = wxString(static_cast<char const*>(dataBuffer.GetData())+sizeof(wxDataFormatId),wxConvLocal);
796 itemStringAvailable = true;
797 }
798 break;
799 case wxDF_UNICODETEXT:
800 {
801 event.GetDataObject()->GetDataHere(wxDF_UNICODETEXT,dataBuffer.GetAppendBuf(dataSize));
802 dataBuffer.UngetAppendBuf(dataSize);
803 if (itemStringAvailable) // does an object already exist as an ASCII text (see wxDF_TEXT case statement)?
804 [dataArray replaceObjectAtIndex:itemCounter withObject:[NSData dataWithBytes:dataBuffer.GetData() length:dataBufferSize]];
805 else
806 [dataArray addObject:[NSData dataWithBytes:dataBuffer.GetData() length:dataBufferSize]];
807 itemString = wxString::FromUTF8(static_cast<char const*>(dataBuffer.GetData())+sizeof(wxDataFormatId),dataSize);
808 itemStringAvailable = true;
809 } /* block */
810 break;
811 default:
812 wxFAIL_MSG("Data object has invalid or unsupported data format");
813 [dataArray release];
814 return NO;
815 }
816 }
817 delete[] dataFormats;
818 delete itemObject;
819 if (dataStringAvailable)
820 if (itemStringAvailable)
821 {
822 if (itemCounter > 0)
823 dataString << wxT('\n');
824 dataString << itemString;
825 }
826 else
827 dataStringAvailable = false;
828 }
829 else
830 {
831 [dataArray release];
832 delete itemObject;
833 return NO; // dragging was vetoed or no data available
834 }
835 }
836 if (dataStringAvailable)
837 {
838 wxCFStringRef osxString(dataString);
839
840 [pasteboard declareTypes:[NSArray arrayWithObjects:DataViewPboardType,NSStringPboardType,nil] owner:nil];
841 [pasteboard setPropertyList:dataArray forType:DataViewPboardType];
842 [pasteboard setString:osxString.AsNSString() forType:NSStringPboardType];
843 }
844 else
845 {
846 [pasteboard declareTypes:[NSArray arrayWithObject:DataViewPboardType] owner:nil];
847 [pasteboard setPropertyList:dataArray forType:DataViewPboardType];
848 }
849 return YES;
850 }
851 else
852 return NO; // no items to drag (should never occur)
853 }
854
855 //
856 // buffer handling
857 //
858 -(void) addToBuffer:(wxPointerObject*)item
859 {
860 [items addObject:item];
861 }
862
863 -(void) clearBuffer
864 {
865 [items removeAllObjects];
866 }
867
868 -(wxPointerObject*) getDataViewItemFromBuffer:(const wxDataViewItem&)item
869 {
870 return [items member:[[[wxPointerObject alloc] initWithPointer:item.GetID()] autorelease]];
871 }
872
873 -(wxPointerObject*) getItemFromBuffer:(wxPointerObject*)item
874 {
875 return [items member:item];
876 }
877
878 -(BOOL) isInBuffer:(wxPointerObject*)item
879 {
880 return [items containsObject:item];
881 }
882
883 -(void) removeFromBuffer:(wxPointerObject*)item
884 {
885 [items removeObject:item];
886 }
887
888 //
889 // children handling
890 //
891 -(void) appendChild:(wxPointerObject*)item
892 {
893 [children addObject:item];
894 }
895
896 -(void) clearChildren
897 {
898 [children removeAllObjects];
899 }
900
901 -(wxPointerObject*) getChild:(NSUInteger)index
902 {
903 return [children objectAtIndex:index];
904 }
905
906 -(NSUInteger) getChildCount
907 {
908 return [children count];
909 }
910
911 -(void) removeChild:(NSUInteger)index
912 {
913 [children removeObjectAtIndex:index];
914 }
915
916 //
917 // buffer handling
918 //
919 -(void) clearBuffers
920 {
921 [self clearBuffer];
922 [self clearChildren];
923 [self setCurrentParentItem:nil];
924 }
925
926 //
927 // sorting
928 //
929 -(NSArray*) sortDescriptors
930 {
931 return sortDescriptors;
932 }
933
934 -(void) setSortDescriptors:(NSArray*)newSortDescriptors
935 {
936 [newSortDescriptors retain];
937 [sortDescriptors release];
938 sortDescriptors = newSortDescriptors;
939 }
940
941 //
942 // access to wxWidget's implementation
943 //
944 -(wxPointerObject*) currentParentItem
945 {
946 return currentParentItem;
947 }
948
949 -(wxCocoaDataViewControl*) implementation
950 {
951 return implementation;
952 }
953
954 -(wxDataViewModel*) model
955 {
956 return model;
957 }
958
959 -(void) setCurrentParentItem:(wxPointerObject*)newCurrentParentItem
960 {
961 [newCurrentParentItem retain];
962 [currentParentItem release];
963 currentParentItem = newCurrentParentItem;
964 }
965
966 -(void) setImplementation:(wxCocoaDataViewControl*) newImplementation
967 {
968 implementation = newImplementation;
969 }
970
971 -(void) setModel:(wxDataViewModel*) newModel
972 {
973 model = newModel;
974 }
975
976 //
977 // other methods
978 //
979 -(void) bufferItem:(wxPointerObject*)parentItem withChildren:(wxDataViewItemArray*)dataViewChildrenPtr
980 {
981 NSInteger const noOfChildren = (*dataViewChildrenPtr).GetCount();
982
983 [self setCurrentParentItem:parentItem];
984 [self clearChildren];
985 for (NSInteger indexChild=0; indexChild<noOfChildren; ++indexChild)
986 {
987 wxPointerObject* bufferedPointerObject;
988 wxPointerObject* newPointerObject([[wxPointerObject alloc] initWithPointer:(*dataViewChildrenPtr)[indexChild].GetID()]);
989
990 // The next statement and test looks strange but there is
991 // unfortunately no workaround: due to the fact that two pointer
992 // objects are identical if their pointers are identical - because the
993 // method isEqual has been overloaded - the set operation will only
994 // add a new pointer object if there is not already one in the set
995 // having the same pointer. On the other side the children's array
996 // would always add the new pointer object. This means that different
997 // pointer objects are stored in the set and array. This will finally
998 // lead to a crash as objects diverge. To solve this issue it is first
999 // tested if the child already exists in the set and if it is the case
1000 // the sets object is going to be appended to the array, otheriwse the
1001 // new pointer object is added to the set and array:
1002 bufferedPointerObject = [self getItemFromBuffer:newPointerObject];
1003 if (bufferedPointerObject == nil)
1004 {
1005 [items addObject:newPointerObject];
1006 [children addObject:newPointerObject];
1007 }
1008 else
1009 [children addObject:bufferedPointerObject];
1010 [newPointerObject release];
1011 }
1012 }
1013
1014 @end
1015
1016 // ============================================================================
1017 // wxCustomCell
1018 // ============================================================================
1019
1020 @implementation wxCustomCell
1021
1022 -(NSSize) cellSize
1023 {
1024 wxCustomRendererObject * const
1025 obj = static_cast<wxCustomRendererObject *>([self objectValue]);
1026
1027
1028 const wxSize size = obj->customRenderer->GetSize();
1029 return NSMakeSize(size.x, size.y);
1030 }
1031
1032 //
1033 // implementations
1034 //
1035 -(void) drawWithFrame:(NSRect)cellFrame inView:(NSView*)controlView
1036 {
1037 wxCustomRendererObject * const
1038 obj = static_cast<wxCustomRendererObject *>([self objectValue]);
1039 wxDataViewCustomRenderer * const renderer = obj->customRenderer;
1040
1041 // draw its own background:
1042 [[self backgroundColor] set];
1043 NSRectFill(cellFrame);
1044
1045 // TODO: attributes support
1046 renderer->Render(wxFromNSRect(controlView, cellFrame), renderer->GetDC(), 0);
1047 renderer->SetDC(NULL);
1048 }
1049
1050 -(NSRect) imageRectForBounds:(NSRect)cellFrame
1051 {
1052 return cellFrame;
1053 }
1054
1055 -(NSRect) titleRectForBounds:(NSRect)cellFrame
1056 {
1057 return cellFrame;
1058 }
1059
1060 @end
1061
1062 // ============================================================================
1063 // wxImageTextCell
1064 // ============================================================================
1065 @implementation wxImageTextCell
1066 //
1067 // initialization
1068 //
1069 -(id) init
1070 {
1071 self = [super init];
1072 if (self != nil)
1073 {
1074 // initializing the text part:
1075 [self setSelectable:YES];
1076 // initializing the image part:
1077 image = nil;
1078 imageSize = NSMakeSize(16,16);
1079 spaceImageText = 5.0;
1080 xImageShift = 5.0;
1081 }
1082 return self;
1083 }
1084
1085 -(id) copyWithZone:(NSZone*)zone
1086 {
1087 wxImageTextCell* cell;
1088
1089
1090 cell = (wxImageTextCell*) [super copyWithZone:zone];
1091 cell->image = [image retain];
1092 cell->imageSize = imageSize;
1093 cell->spaceImageText = spaceImageText;
1094 cell->xImageShift = xImageShift;
1095
1096 return cell;
1097 }
1098
1099 -(void) dealloc
1100 {
1101 [image release];
1102
1103 [super dealloc];
1104 }
1105
1106 //
1107 // alignment
1108 //
1109 -(NSTextAlignment) alignment
1110 {
1111 return cellAlignment;
1112 }
1113
1114 -(void) setAlignment:(NSTextAlignment)newAlignment
1115 {
1116 cellAlignment = newAlignment;
1117 switch (newAlignment)
1118 {
1119 case NSCenterTextAlignment:
1120 case NSLeftTextAlignment:
1121 case NSJustifiedTextAlignment:
1122 case NSNaturalTextAlignment:
1123 [super setAlignment:NSLeftTextAlignment];
1124 break;
1125 case NSRightTextAlignment:
1126 [super setAlignment:NSRightTextAlignment];
1127 break;
1128 default:
1129 wxFAIL_MSG("Unknown alignment type.");
1130 }
1131 }
1132
1133 //
1134 // image access
1135 //
1136 -(NSImage*) image
1137 {
1138 return image;
1139 }
1140
1141 -(void) setImage:(NSImage*)newImage
1142 {
1143 [newImage retain];
1144 [image release];
1145 image = newImage;
1146 }
1147
1148 -(NSSize) imageSize
1149 {
1150 return imageSize;
1151 }
1152
1153 -(void) setImageSize:(NSSize) newImageSize
1154 {
1155 imageSize = newImageSize;
1156 }
1157
1158 //
1159 // other methods
1160 //
1161 -(NSSize) cellImageSize
1162 {
1163 return NSMakeSize(imageSize.width+xImageShift+spaceImageText,imageSize.height);
1164 }
1165
1166 -(NSSize) cellSize
1167 {
1168 NSSize cellSize([super cellSize]);
1169
1170
1171 if (imageSize.height > cellSize.height)
1172 cellSize.height = imageSize.height;
1173 cellSize.width += imageSize.width+xImageShift+spaceImageText;
1174
1175 return cellSize;
1176 }
1177
1178 -(NSSize) cellTextSize
1179 {
1180 return [super cellSize];
1181 }
1182
1183 //
1184 // implementations
1185 //
1186 -(void) determineCellParts:(NSRect)cellFrame imagePart:(NSRect*)imageFrame textPart:(NSRect*)textFrame
1187 {
1188 switch (cellAlignment)
1189 {
1190 case NSCenterTextAlignment:
1191 {
1192 CGFloat const cellSpace = cellFrame.size.width-[self cellSize].width;
1193
1194 if (cellSpace <= 0) // if the cell's frame is smaller than its contents (at least in x-direction) make sure that the image is visible:
1195 NSDivideRect(cellFrame,imageFrame,textFrame,xImageShift+imageSize.width+spaceImageText,NSMinXEdge);
1196 else // otherwise center the image and text in the cell's frame
1197 NSDivideRect(cellFrame,imageFrame,textFrame,xImageShift+imageSize.width+spaceImageText+0.5*cellSpace,NSMinXEdge);
1198 }
1199 break;
1200 case NSJustifiedTextAlignment:
1201 case NSLeftTextAlignment:
1202 case NSNaturalTextAlignment: // how to determine the natural writing direction? TODO
1203 NSDivideRect(cellFrame,imageFrame,textFrame,xImageShift+imageSize.width+spaceImageText,NSMinXEdge);
1204 break;
1205 case NSRightTextAlignment:
1206 {
1207 CGFloat const cellSpace = cellFrame.size.width-[self cellSize].width;
1208
1209 if (cellSpace <= 0) // if the cell's frame is smaller than its contents (at least in x-direction) make sure that the image is visible:
1210 NSDivideRect(cellFrame,imageFrame,textFrame,xImageShift+imageSize.width+spaceImageText,NSMinXEdge);
1211 else // otherwise right align the image and text in the cell's frame
1212 NSDivideRect(cellFrame,imageFrame,textFrame,xImageShift+imageSize.width+spaceImageText+cellSpace,NSMinXEdge);
1213 }
1214 break;
1215 default:
1216 *imageFrame = NSZeroRect;
1217 *textFrame = NSZeroRect;
1218 wxFAIL_MSG("Unhandled alignment type.");
1219 }
1220 }
1221
1222 -(void) drawWithFrame:(NSRect)cellFrame inView:(NSView*)controlView
1223 {
1224 NSRect textFrame, imageFrame;
1225
1226
1227 [self determineCellParts:cellFrame imagePart:&imageFrame textPart:&textFrame];
1228 // draw the image part by ourselves;
1229 // check if the cell has to draw its own background (checking is done by the parameter of the textfield's cell):
1230 if ([self drawsBackground])
1231 {
1232 [[self backgroundColor] set];
1233 NSRectFill(imageFrame);
1234 }
1235 if (image != nil)
1236 {
1237 // the image is slightly shifted (xImageShift) and has a fixed size but the image's frame might be larger and starts
1238 // currently on the left side of the cell's frame; therefore, the origin and the image's frame size have to be adjusted:
1239 if (imageFrame.size.width >= xImageShift+imageSize.width+spaceImageText)
1240 {
1241 imageFrame.origin.x += imageFrame.size.width-imageSize.width-spaceImageText;
1242 imageFrame.size.width = imageSize.width;
1243 }
1244 else
1245 {
1246 imageFrame.origin.x += xImageShift;
1247 imageFrame.size.width -= xImageShift+spaceImageText;
1248 }
1249 // ...and the image has to be centered in the y-direction:
1250 if (imageFrame.size.height > imageSize.height)
1251 imageFrame.size.height = imageSize.height;
1252 imageFrame.origin.y += ceil(0.5*(cellFrame.size.height-imageFrame.size.height));
1253
1254 // according to the documentation the coordinate system should be flipped for NSTableViews (y-coordinate goes from top to bottom);
1255 // to draw an image correctly the coordinate system has to be transformed to a bottom-top coordinate system, otherwise the image's
1256 // content is flipped:
1257 NSAffineTransform* coordinateTransform([NSAffineTransform transform]);
1258
1259 if ([controlView isFlipped])
1260 {
1261 [coordinateTransform scaleXBy: 1.0 yBy:-1.0]; // first the coordinate system is brought back to bottom-top orientation
1262 [coordinateTransform translateXBy:0.0 yBy:(-2.0)*imageFrame.origin.y-imageFrame.size.height]; // the coordinate system has to be moved to compensate for the
1263 [coordinateTransform concat]; // other orientation and the position of the image's frame
1264 }
1265 [image drawInRect:imageFrame fromRect:NSZeroRect operation:NSCompositeSourceOver fraction:1.0]; // suggested method to draw the image
1266 // instead of compositeToPoint:operation:
1267 // take back previous transformation (if the view is not flipped the coordinate transformation matrix contains the identity matrix
1268 // and the next two operations do not change the content's transformation matrix):
1269 [coordinateTransform invert];
1270 [coordinateTransform concat];
1271 }
1272 // let the textfield cell draw the text part:
1273 if (textFrame.size.width > [self cellTextSize].width) // for unknown reasons the alignment of the text cell is ignored; therefore change the size so that
1274 textFrame.size.width = [self cellTextSize].width; // alignment does not influence the visualization anymore
1275 [super drawWithFrame:textFrame inView:controlView];
1276 }
1277
1278 -(void) editWithFrame:(NSRect)aRect inView:(NSView*)controlView editor:(NSText*)textObj delegate:(id)anObject event:(NSEvent*)theEvent
1279 {
1280 NSRect textFrame, imageFrame;
1281
1282
1283 [self determineCellParts:aRect imagePart:&imageFrame textPart:&textFrame];
1284 [super editWithFrame:textFrame inView:controlView editor:textObj delegate:anObject event:theEvent];
1285 }
1286
1287 #if MAC_OS_X_VERSION_MAX_ALLOWED >= MAC_OS_X_VERSION_10_5
1288 -(NSUInteger) hitTestForEvent:(NSEvent*)event inRect:(NSRect)cellFrame ofView:(NSView*)controlView
1289 {
1290 NSPoint point = [controlView convertPoint:[event locationInWindow] fromView:nil];
1291
1292 NSRect imageFrame, textFrame;
1293
1294
1295 [self determineCellParts:cellFrame imagePart:&imageFrame textPart:&textFrame];
1296 if (image != nil)
1297 {
1298 // the image is shifted...
1299 if (imageFrame.size.width >= xImageShift+imageSize.width+spaceImageText)
1300 {
1301 imageFrame.origin.x += imageFrame.size.width-imageSize.width-spaceImageText;
1302 imageFrame.size.width = imageSize.width;
1303 }
1304 else
1305 {
1306 imageFrame.origin.x += xImageShift;
1307 imageFrame.size.width -= xImageShift+spaceImageText;
1308 }
1309 // ...and centered:
1310 if (imageFrame.size.height > imageSize.height)
1311 imageFrame.size.height = imageSize.height;
1312 imageFrame.origin.y += ceil(0.5*(cellFrame.size.height-imageFrame.size.height));
1313 // If the point is in the image rect, then it is a content hit (see documentation for hitTestForEvent:inRect:ofView):
1314 if (NSMouseInRect(point, imageFrame, [controlView isFlipped]))
1315 return NSCellHitContentArea;
1316 }
1317 // if the image was not hit let's try the text part:
1318 if (textFrame.size.width > [self cellTextSize].width) // for unknown reasons the alignment of the text cell is ignored; therefore change the size so that
1319 textFrame.size.width = [self cellTextSize].width; // alignment does not influence the visualization anymore
1320 return [super hitTestForEvent:event inRect:textFrame ofView:controlView];
1321 }
1322 #endif
1323
1324 -(NSRect) imageRectForBounds:(NSRect)cellFrame
1325 {
1326 NSRect textFrame, imageFrame;
1327
1328
1329 [self determineCellParts:cellFrame imagePart:&imageFrame textPart:&textFrame];
1330 if (imageFrame.size.width >= xImageShift+imageSize.width+spaceImageText)
1331 {
1332 imageFrame.origin.x += imageFrame.size.width-imageSize.width-spaceImageText;
1333 imageFrame.size.width = imageSize.width;
1334 }
1335 else
1336 {
1337 imageFrame.origin.x += xImageShift;
1338 imageFrame.size.width -= xImageShift+spaceImageText;
1339 }
1340 // ...and centered:
1341 if (imageFrame.size.height > imageSize.height)
1342 imageFrame.size.height = imageSize.height;
1343 imageFrame.origin.y += ceil(0.5*(cellFrame.size.height-imageFrame.size.height));
1344
1345 return imageFrame;
1346 }
1347
1348 -(void) selectWithFrame:(NSRect)aRect inView:(NSView*)controlView editor:(NSText*)textObj delegate:(id)anObject start:(NSInteger)selStart length:(NSInteger)selLength
1349 {
1350 NSRect textFrame, imageFrame;
1351
1352
1353 [self determineCellParts:aRect imagePart:&imageFrame textPart:&textFrame];
1354 [super selectWithFrame:textFrame inView:controlView editor:textObj delegate:anObject start:selStart length:selLength];
1355 }
1356
1357 -(NSRect) titleRectForBounds:(NSRect)cellFrame
1358 {
1359 NSRect textFrame, imageFrame;
1360
1361
1362 [self determineCellParts:cellFrame imagePart:&imageFrame textPart:&textFrame];
1363 return textFrame;
1364 }
1365
1366 @end
1367
1368 // ============================================================================
1369 // wxCocoaOutlineView
1370 // ============================================================================
1371 @implementation wxCocoaOutlineView
1372
1373 //
1374 // initializers / destructor
1375 //
1376 -(id) init
1377 {
1378 self = [super init];
1379 if (self != nil)
1380 {
1381 currentlyEditedColumn =
1382 currentlyEditedRow = -1;
1383
1384 [self registerForDraggedTypes:[NSArray arrayWithObjects:DataViewPboardType,NSStringPboardType,nil]];
1385 [self setDelegate:self];
1386 [self setDoubleAction:@selector(actionDoubleClick:)];
1387 [self setDraggingSourceOperationMask:NSDragOperationEvery forLocal:NO];
1388 [self setDraggingSourceOperationMask:NSDragOperationEvery forLocal:YES];
1389 [self setTarget:self];
1390 }
1391 return self;
1392 }
1393
1394 //
1395 // access to wxWidget's implementation
1396 //
1397 -(wxCocoaDataViewControl*) implementation
1398 {
1399 return implementation;
1400 }
1401
1402 -(void) setImplementation:(wxCocoaDataViewControl*) newImplementation
1403 {
1404 implementation = newImplementation;
1405 }
1406
1407 //
1408 // actions
1409 //
1410 -(void) actionDoubleClick:(id)sender
1411 // actually the documentation (NSTableView 2007-10-31) for doubleAction: and setDoubleAction: seems to be wrong as this action message is always sent
1412 // whether the cell is editable or not
1413 {
1414 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1415
1416 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_ACTIVATED,dvc->GetId()); // variable definition
1417
1418
1419 event.SetEventObject(dvc);
1420 event.SetItem(wxDataViewItem([((wxPointerObject*) [self itemAtRow:[self clickedRow]]) pointer]));
1421 dvc->GetEventHandler()->ProcessEvent(event);
1422 }
1423
1424
1425 //
1426 // contextual menus
1427 //
1428 -(NSMenu*) menuForEvent:(NSEvent*)theEvent
1429 // this method does not do any special menu event handling but only sends an event message; therefore, the user
1430 // has full control if a context menu should be shown or not
1431 {
1432 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1433
1434 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_CONTEXT_MENU,dvc->GetId());
1435
1436 wxDataViewItemArray selectedItems;
1437
1438
1439 event.SetEventObject(dvc);
1440 event.SetModel(dvc->GetModel());
1441 // get the item information;
1442 // theoretically more than one ID can be returned but the event can only handle one item, therefore only the first
1443 // item of the array is returned:
1444 if (dvc->GetSelections(selectedItems) > 0)
1445 event.SetItem(selectedItems[0]);
1446 dvc->GetEventHandler()->ProcessEvent(event);
1447 // nothing is done:
1448 return nil;
1449 }
1450
1451 //
1452 // delegate methods
1453 //
1454 -(void) outlineView:(NSOutlineView*)outlineView mouseDownInHeaderOfTableColumn:(NSTableColumn*)tableColumn
1455 {
1456 wxDataViewColumn* const col(static_cast<wxDataViewColumn*>([[tableColumn identifier] pointer]));
1457
1458 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1459
1460 wxDataViewEvent
1461 event(wxEVT_COMMAND_DATAVIEW_COLUMN_HEADER_CLICK,dvc->GetId());
1462
1463
1464 // first, send an event that the user clicked into a column's header:
1465 event.SetEventObject(dvc);
1466 event.SetColumn(dvc->GetColumnPosition(col));
1467 event.SetDataViewColumn(col);
1468 dvc->HandleWindowEvent(event);
1469
1470 // now, check if the click may have had an influence on sorting, too;
1471 // the sorting setup has to be done only if the clicked table column is sortable and has not been used for
1472 // sorting before the click; if the column is already responsible for sorting the native control changes
1473 // the sorting direction automatically and informs the data source via outlineView:sortDescriptorsDidChange:
1474 if (col->IsSortable() && ([tableColumn sortDescriptorPrototype] == nil))
1475 {
1476 // remove the sort order from the previously sorted column table (it can also be that
1477 // no sorted column table exists):
1478 UInt32 const noOfColumns = [outlineView numberOfColumns];
1479
1480 for (UInt32 i=0; i<noOfColumns; ++i)
1481 [[[outlineView tableColumns] objectAtIndex:i] setSortDescriptorPrototype:nil];
1482 // make column table sortable:
1483 NSArray* sortDescriptors;
1484 NSSortDescriptor* sortDescriptor;
1485
1486 sortDescriptor = [[NSSortDescriptor alloc] initWithKey:[NSString stringWithFormat:@"%d",[outlineView columnWithIdentifier:[tableColumn identifier]]]
1487 ascending:YES];
1488 sortDescriptors = [NSArray arrayWithObject:sortDescriptor];
1489 [tableColumn setSortDescriptorPrototype:sortDescriptor];
1490 [outlineView setSortDescriptors:sortDescriptors];
1491 [sortDescriptor release];
1492 }
1493 }
1494
1495 -(BOOL) outlineView:(NSOutlineView*)outlineView shouldCollapseItem:(id)item
1496 {
1497 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1498
1499 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_COLLAPSING,dvc->GetId()); // variable definition
1500
1501
1502 event.SetEventObject(dvc);
1503 event.SetItem (wxDataViewItem([((wxPointerObject*) item) pointer]));
1504 event.SetModel (dvc->GetModel());
1505 // finally send the equivalent wxWidget event:
1506 dvc->GetEventHandler()->ProcessEvent(event);
1507 // opening the container is allowed if not vetoed:
1508 return event.IsAllowed();
1509 }
1510
1511 -(BOOL) outlineView:(NSOutlineView*)outlineView shouldExpandItem:(id)item
1512 {
1513 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1514
1515 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_EXPANDING,dvc->GetId()); // variable definition
1516
1517
1518 event.SetEventObject(dvc);
1519 event.SetItem (wxDataViewItem([((wxPointerObject*) item) pointer]));
1520 event.SetModel (dvc->GetModel());
1521 // finally send the equivalent wxWidget event:
1522 dvc->GetEventHandler()->ProcessEvent(event);
1523 // opening the container is allowed if not vetoed:
1524 return event.IsAllowed();
1525 }
1526
1527 -(BOOL) outlineView:(NSOutlineView*)outlineView shouldSelectTableColumn:(NSTableColumn*)tableColumn
1528 {
1529 return NO;
1530 }
1531
1532 -(void) outlineView:(wxCocoaOutlineView*)outlineView
1533 willDisplayCell:(id)cell
1534 forTableColumn:(NSTableColumn*)tableColumn
1535 item:(id)item
1536 {
1537 wxDataViewCtrl * const dvc = implementation->GetDataViewCtrl();
1538 wxDataViewModel * const model = dvc->GetModel();
1539
1540 wxDataViewColumn * const
1541 dvCol(static_cast<wxDataViewColumn*>(
1542 [[tableColumn identifier] pointer]
1543 )
1544 );
1545
1546 wxDataViewRenderer * const renderer = dvCol->GetRenderer();
1547 wxDataViewRendererNativeData * const data = renderer->GetNativeData();
1548
1549 wxDataViewItem dvItem([static_cast<wxPointerObject *>(item) pointer]);
1550
1551 // set the font and text colour to use: we need to do it if we had ever
1552 // changed them before, even if this item itself doesn't have any special
1553 // attributes as otherwise it would reuse the attributes from the previous
1554 // cell rendered using the same renderer
1555 NSFont *font = NULL;
1556 NSColor *colText = NULL;
1557
1558 wxDataViewItemAttr attr;
1559 if ( model && model->GetAttr(dvItem, dvCol->GetModelColumn(), attr) )
1560 {
1561 if ( attr.HasFont() )
1562 {
1563 font = data->GetOriginalFont();
1564 if ( !font )
1565 {
1566 // this is the first time we're setting the font, remember the
1567 // original one before changing it
1568 font = [cell font];
1569 data->SaveOriginalFont(font);
1570 }
1571
1572 if ( font )
1573 {
1574 // FIXME: using wxFont methods here doesn't work for some reason
1575 NSFontManager * const fm = [NSFontManager sharedFontManager];
1576 if ( attr.GetBold() )
1577 font = [fm convertFont:font toHaveTrait:NSBoldFontMask];
1578 if ( attr.GetItalic() )
1579 font = [fm convertFont:font toHaveTrait:NSItalicFontMask];
1580 }
1581 //else: can't change font if the cell doesn't have any
1582 }
1583
1584 if ( attr.HasColour() )
1585 {
1586 // we can set font for any cell but only NSTextFieldCell provides
1587 // a method for setting text colour so check that this method is
1588 // available before using it
1589 if ( [cell respondsToSelector:@selector(setTextColor:)] &&
1590 [cell respondsToSelector:@selector(textColor)] )
1591 {
1592 if ( !data->GetOriginalTextColour() )
1593 {
1594 data->SaveOriginalTextColour([cell textColor]);
1595 }
1596
1597 const wxColour& c = attr.GetColour();
1598 colText = [NSColor colorWithDeviceRed:c.Red() / 255.
1599 green:c.Green() / 255.
1600 blue:c.Blue() / 255.
1601 alpha:c.Alpha() / 255.];
1602 }
1603 }
1604 }
1605
1606 if ( !font )
1607 font = data->GetOriginalFont();
1608 if ( !colText )
1609 colText = data->GetOriginalTextColour();
1610
1611 if ( font )
1612 [cell setFont:font];
1613
1614 if ( colText )
1615 [cell setTextColor:colText];
1616
1617
1618 data->SetColumnPtr(tableColumn);
1619 data->SetItem(item);
1620 data->SetItemCell(cell);
1621
1622 renderer->MacRender();
1623 }
1624
1625 //
1626 // notifications
1627 //
1628 -(void) outlineViewColumnDidMove:(NSNotification*)notification
1629 {
1630 int const newColumnPosition = [[[notification userInfo] objectForKey:@"NSNewColumn"] intValue];
1631
1632 wxDataViewColumn* const col(static_cast<wxDataViewColumn*>([[[[self tableColumns] objectAtIndex:newColumnPosition] identifier] pointer]));
1633
1634 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1635
1636 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_COLUMN_REORDERED,dvc->GetId());
1637
1638
1639 event.SetEventObject(dvc);
1640 event.SetColumn(dvc->GetColumnPosition(col));
1641 event.SetDataViewColumn(col);
1642 dvc->GetEventHandler()->ProcessEvent(event);
1643 }
1644
1645 -(void) outlineViewItemDidCollapse:(NSNotification*)notification
1646 {
1647 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1648
1649 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_COLLAPSED,dvc->GetId());
1650
1651
1652 event.SetEventObject(dvc);
1653 event.SetItem(wxDataViewItem([((wxPointerObject*) [[notification userInfo] objectForKey:@"NSObject"]) pointer]));
1654 dvc->GetEventHandler()->ProcessEvent(event);
1655 }
1656
1657 -(void) outlineViewItemDidExpand:(NSNotification*)notification
1658 {
1659 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1660
1661 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_ITEM_EXPANDED,dvc->GetId());
1662
1663
1664 event.SetEventObject(dvc);
1665 event.SetItem(wxDataViewItem([((wxPointerObject*) [[notification userInfo] objectForKey:@"NSObject"]) pointer]));
1666 dvc->GetEventHandler()->ProcessEvent(event);
1667 }
1668
1669 -(void) outlineViewSelectionDidChange:(NSNotification*)notification
1670 {
1671 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1672
1673 wxDataViewEvent event(wxEVT_COMMAND_DATAVIEW_SELECTION_CHANGED,dvc->GetId()); // variable definition
1674
1675
1676 event.SetEventObject(dvc);
1677 event.SetModel (dvc->GetModel());
1678 dvc->GetEventHandler()->ProcessEvent(event);
1679 }
1680
1681 -(void) textDidBeginEditing:(NSNotification*)notification
1682 // this notification is only sent if the user started modifying the cell (not when the user clicked into the cell
1683 // and the cell's editor is called!)
1684 {
1685 // call method of superclass (otherwise editing does not work correctly - the outline data source class is not
1686 // informed about a change of data):
1687 [super textDidBeginEditing:notification];
1688
1689 // remember the column being edited, it will be used in textDidEndEditing:
1690 currentlyEditedColumn = [self editedColumn];
1691 currentlyEditedRow = [self editedRow];
1692
1693 wxDataViewColumn* const col =
1694 static_cast<wxDataViewColumn*>(
1695 [[[[self tableColumns] objectAtIndex:currentlyEditedColumn] identifier] pointer]);
1696
1697 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1698
1699
1700 // stop editing of a custom item first (if necessary)
1701 dvc->FinishCustomItemEditing();
1702
1703 // now, send the event:
1704 wxDataViewEvent
1705 event(wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_STARTED,dvc->GetId()); // variable definition
1706
1707 event.SetEventObject(dvc);
1708 event.SetItem(
1709 wxDataViewItem([((wxPointerObject*) [self itemAtRow:currentlyEditedRow]) pointer]));
1710 event.SetColumn(dvc->GetColumnPosition(col));
1711 event.SetDataViewColumn(col);
1712 dvc->GetEventHandler()->ProcessEvent(event);
1713 }
1714
1715 -(void) textDidEndEditing:(NSNotification*)notification
1716 {
1717 // call method of superclass (otherwise editing does not work correctly - the outline data source class is not
1718 // informed about a change of data):
1719 [super textDidEndEditing:notification];
1720
1721 // under OSX an event indicating the end of an editing session can be sent even if no event indicating a start of an
1722 // editing session has been sent (see Documentation for NSControl controlTextDidEndEditing:); this is not expected by a user
1723 // of the wxWidgets library and therefore an wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_DONE event is only sent if a corresponding
1724 // wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_STARTED has been sent before; to check if a wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_STARTED
1725 // has been sent the last edited column/row are valid:
1726 if ( currentlyEditedColumn != -1 && currentlyEditedRow != -1 )
1727 {
1728 wxDataViewColumn* const col =
1729 static_cast<wxDataViewColumn*>(
1730 [[[[self tableColumns] objectAtIndex:currentlyEditedColumn] identifier] pointer]);
1731
1732 wxDataViewCtrl* const dvc = implementation->GetDataViewCtrl();
1733
1734 // send event to wxWidgets:
1735 wxDataViewEvent
1736 event(wxEVT_COMMAND_DATAVIEW_ITEM_EDITING_DONE,dvc->GetId()); // variable definition
1737
1738 event.SetEventObject(dvc);
1739 event.SetItem(
1740 wxDataViewItem([((wxPointerObject*) [self itemAtRow:currentlyEditedRow]) pointer]));
1741 event.SetColumn(dvc->GetColumnPosition(col));
1742 event.SetDataViewColumn(col);
1743 dvc->GetEventHandler()->ProcessEvent(event);
1744
1745
1746 // we're not editing any more
1747 currentlyEditedColumn =
1748 currentlyEditedRow = -1;
1749 }
1750 }
1751
1752 @end
1753 // ============================================================================
1754 // wxCocoaDataViewControl
1755 // ============================================================================
1756 //
1757 // constructors / destructor
1758 //
1759 wxCocoaDataViewControl::wxCocoaDataViewControl(wxWindow* peer, const wxPoint& pos, const wxSize& size, long style)
1760 :wxWidgetCocoaImpl(peer,[[NSScrollView alloc] initWithFrame:wxOSXGetFrameForControl(peer,pos,size)]),
1761 m_DataSource(NULL), m_OutlineView([[wxCocoaOutlineView alloc] init])
1762 {
1763 // initialize scrollview (the outline view is part of a scrollview):
1764 NSScrollView* scrollview = (NSScrollView*) GetWXWidget(); // definition for abbreviational purposes
1765
1766
1767 [scrollview setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable];
1768 [scrollview setBorderType:NSNoBorder];
1769 [scrollview setHasVerticalScroller:YES];
1770 [scrollview setHasHorizontalScroller:YES];
1771 [scrollview setAutohidesScrollers:YES];
1772 [scrollview setDocumentView:m_OutlineView];
1773
1774 // setting up the native control itself
1775 NSUInteger maskGridStyle(NSTableViewGridNone);
1776
1777 [m_OutlineView setImplementation:this];
1778 [m_OutlineView setColumnAutoresizingStyle:NSTableViewSequentialColumnAutoresizingStyle];
1779 [m_OutlineView setIndentationPerLevel:GetDataViewCtrl()->GetIndent()];
1780 if (style & wxDV_HORIZ_RULES)
1781 maskGridStyle |= NSTableViewSolidHorizontalGridLineMask;
1782 if (style & wxDV_VERT_RULES)
1783 maskGridStyle |= NSTableViewSolidVerticalGridLineMask;
1784 [m_OutlineView setGridStyleMask:maskGridStyle];
1785 [m_OutlineView setAllowsMultipleSelection: (style & wxDV_MULTIPLE) != 0];
1786 [m_OutlineView setUsesAlternatingRowBackgroundColors:(style & wxDV_ROW_LINES) != 0];
1787 }
1788
1789 wxCocoaDataViewControl::~wxCocoaDataViewControl()
1790 {
1791 [m_DataSource release];
1792 [m_OutlineView release];
1793 }
1794
1795 //
1796 // column related methods (inherited from wxDataViewWidgetImpl)
1797 //
1798 bool wxCocoaDataViewControl::ClearColumns()
1799 {
1800 bool const bufAllowsMultipleSelection = [m_OutlineView allowsMultipleSelection];
1801
1802
1803 // as there is a bug in NSOutlineView version (OSX 10.5.6 #6555162) the columns cannot be deleted if there is an outline column in the view;
1804 // therefore, the whole view is deleted and newly constructed:
1805 [m_OutlineView release];
1806 m_OutlineView = [[wxCocoaOutlineView alloc] init];
1807 [((NSScrollView*) GetWXWidget()) setDocumentView:m_OutlineView];
1808
1809 // setting up the native control itself
1810 [m_OutlineView setImplementation:this];
1811 [m_OutlineView setColumnAutoresizingStyle:NSTableViewSequentialColumnAutoresizingStyle];
1812 [m_OutlineView setIndentationPerLevel:GetDataViewCtrl()->GetIndent()];
1813 if (bufAllowsMultipleSelection)
1814 [m_OutlineView setAllowsMultipleSelection:YES];
1815 [m_OutlineView setDataSource:m_DataSource];
1816 // done:
1817 return true;
1818 }
1819
1820 bool wxCocoaDataViewControl::DeleteColumn(wxDataViewColumn* columnPtr)
1821 {
1822 if ([m_OutlineView outlineTableColumn] == columnPtr->GetNativeData()->GetNativeColumnPtr())
1823 [m_OutlineView setOutlineTableColumn:nil]; // due to a bug this does not work
1824 [m_OutlineView removeTableColumn:columnPtr->GetNativeData()->GetNativeColumnPtr()]; // due to a confirmed bug #6555162 the deletion does not work for
1825 // outline table columns (... and there is no workaround)
1826 return (([m_OutlineView columnWithIdentifier:[[[wxPointerObject alloc] initWithPointer:columnPtr] autorelease]]) == -1);
1827 }
1828
1829 void wxCocoaDataViewControl::DoSetExpanderColumn(const wxDataViewColumn *columnPtr)
1830 {
1831 [m_OutlineView setOutlineTableColumn:columnPtr->GetNativeData()->GetNativeColumnPtr()];
1832 }
1833
1834 wxDataViewColumn* wxCocoaDataViewControl::GetColumn(unsigned int pos) const
1835 {
1836 return static_cast<wxDataViewColumn*>([[[[m_OutlineView tableColumns] objectAtIndex:pos] identifier] pointer]);
1837 }
1838
1839 int wxCocoaDataViewControl::GetColumnPosition(const wxDataViewColumn *columnPtr) const
1840 {
1841 return [m_OutlineView columnWithIdentifier:[[[wxPointerObject alloc] initWithPointer:const_cast<wxDataViewColumn*>(columnPtr)] autorelease]];
1842 }
1843
1844 bool wxCocoaDataViewControl::InsertColumn(unsigned int pos, wxDataViewColumn* columnPtr)
1845 {
1846 // create column and set the native data of the dataview column:
1847 NSTableColumn *nativeColumn = ::CreateNativeColumn(columnPtr);
1848 columnPtr->GetNativeData()->SetNativeColumnPtr(nativeColumn);
1849 // as the native control does not allow the insertion of a column at a specified position the column is first appended and
1850 // - if necessary - moved to its final position:
1851 [m_OutlineView addTableColumn:nativeColumn];
1852 if (pos != static_cast<unsigned int>([m_OutlineView numberOfColumns]-1))
1853 [m_OutlineView moveColumn:[m_OutlineView numberOfColumns]-1 toColumn:pos];
1854 // done:
1855 return true;
1856 }
1857
1858 //
1859 // item related methods (inherited from wxDataViewWidgetImpl)
1860 //
1861 bool wxCocoaDataViewControl::Add(const wxDataViewItem& parent, const wxDataViewItem& WXUNUSED(item))
1862 {
1863 if (parent.IsOk())
1864 [m_OutlineView reloadItem:[m_DataSource getDataViewItemFromBuffer:parent] reloadChildren:YES];
1865 else
1866 [m_OutlineView reloadData];
1867 return true;
1868 }
1869
1870 bool wxCocoaDataViewControl::Add(const wxDataViewItem& parent, const wxDataViewItemArray& WXUNUSED(items))
1871 {
1872 if (parent.IsOk())
1873 [m_OutlineView reloadItem:[m_DataSource getDataViewItemFromBuffer:parent] reloadChildren:YES];
1874 else
1875 [m_OutlineView reloadData];
1876 return true;
1877 }
1878
1879 void wxCocoaDataViewControl::Collapse(const wxDataViewItem& item)
1880 {
1881 [m_OutlineView collapseItem:[m_DataSource getDataViewItemFromBuffer:item]];
1882 }
1883
1884 void wxCocoaDataViewControl::EnsureVisible(const wxDataViewItem& item, const wxDataViewColumn *columnPtr)
1885 {
1886 if (item.IsOk())
1887 {
1888 [m_OutlineView scrollRowToVisible:[m_OutlineView rowForItem:[m_DataSource getDataViewItemFromBuffer:item]]];
1889 if (columnPtr)
1890 [m_OutlineView scrollColumnToVisible:GetColumnPosition(columnPtr)];
1891 }
1892 }
1893
1894 void wxCocoaDataViewControl::Expand(const wxDataViewItem& item)
1895 {
1896 [m_OutlineView expandItem:[m_DataSource getDataViewItemFromBuffer:item]];
1897 }
1898
1899 unsigned int wxCocoaDataViewControl::GetCount() const
1900 {
1901 return [m_OutlineView numberOfRows];
1902 }
1903
1904 wxRect wxCocoaDataViewControl::GetRectangle(const wxDataViewItem& item, const wxDataViewColumn *columnPtr)
1905 {
1906 return wxFromNSRect([m_osxView superview],[m_OutlineView frameOfCellAtColumn:GetColumnPosition(columnPtr)
1907 row:[m_OutlineView rowForItem:[m_DataSource getDataViewItemFromBuffer:item]]]);
1908 }
1909
1910 bool wxCocoaDataViewControl::IsExpanded(const wxDataViewItem& item) const
1911 {
1912 return [m_OutlineView isItemExpanded:[m_DataSource getDataViewItemFromBuffer:item]];
1913 }
1914
1915 bool wxCocoaDataViewControl::Reload()
1916 {
1917 [m_DataSource clearBuffers];
1918 [m_OutlineView scrollColumnToVisible:0];
1919 [m_OutlineView scrollRowToVisible:0];
1920 [m_OutlineView reloadData];
1921 return true;
1922 }
1923
1924 bool wxCocoaDataViewControl::Remove(const wxDataViewItem& parent, const wxDataViewItem& WXUNUSED(item))
1925 {
1926 if (parent.IsOk())
1927 [m_OutlineView reloadItem:[m_DataSource getDataViewItemFromBuffer:parent] reloadChildren:YES];
1928 else
1929 [m_OutlineView reloadData];
1930 return true;
1931 }
1932
1933 bool wxCocoaDataViewControl::Remove(const wxDataViewItem& parent, const wxDataViewItemArray& WXUNUSED(item))
1934 {
1935 if (parent.IsOk())
1936 [m_OutlineView reloadItem:[m_DataSource getDataViewItemFromBuffer:parent] reloadChildren:YES];
1937 else
1938 [m_OutlineView reloadData];
1939 return true;
1940 }
1941
1942 bool wxCocoaDataViewControl::Update(const wxDataViewColumn *columnPtr)
1943 {
1944 return false;
1945 }
1946
1947 bool wxCocoaDataViewControl::Update(const wxDataViewItem& WXUNUSED(parent), const wxDataViewItem& item)
1948 {
1949 [m_OutlineView reloadItem:[m_DataSource getDataViewItemFromBuffer:item]];
1950 return true;
1951 }
1952
1953 bool wxCocoaDataViewControl::Update(const wxDataViewItem& WXUNUSED(parent), const wxDataViewItemArray& items)
1954 {
1955 for (size_t i=0; i<items.GetCount(); ++i)
1956 [m_OutlineView reloadItem:[m_DataSource getDataViewItemFromBuffer:items[i]]];
1957 return true;
1958 }
1959
1960 //
1961 // model related methods
1962 //
1963 bool wxCocoaDataViewControl::AssociateModel(wxDataViewModel* model)
1964 {
1965 [m_DataSource release];
1966 if (model)
1967 {
1968 m_DataSource = [[wxCocoaOutlineDataSource alloc] init];
1969 [m_DataSource setImplementation:this];
1970 [m_DataSource setModel:model];
1971 }
1972 else
1973 m_DataSource = NULL;
1974 [m_OutlineView setDataSource:m_DataSource]; // if there is a data source the data is immediately going to be requested
1975 return true;
1976 }
1977
1978 //
1979 // selection related methods (inherited from wxDataViewWidgetImpl)
1980 //
1981 int wxCocoaDataViewControl::GetSelections(wxDataViewItemArray& sel) const
1982 {
1983 NSIndexSet* selectedRowIndexes([m_OutlineView selectedRowIndexes]);
1984
1985 NSUInteger indexRow;
1986
1987
1988 sel.Empty();
1989 sel.Alloc([selectedRowIndexes count]);
1990 indexRow = [selectedRowIndexes firstIndex];
1991 while (indexRow != NSNotFound)
1992 {
1993 sel.Add(wxDataViewItem([[m_OutlineView itemAtRow:indexRow] pointer]));
1994 indexRow = [selectedRowIndexes indexGreaterThanIndex:indexRow];
1995 }
1996 return sel.GetCount();
1997 }
1998
1999 bool wxCocoaDataViewControl::IsSelected(const wxDataViewItem& item) const
2000 {
2001 return [m_OutlineView isRowSelected:[m_OutlineView rowForItem:[m_DataSource getDataViewItemFromBuffer:item]]];
2002 }
2003
2004 void wxCocoaDataViewControl::Select(const wxDataViewItem& item)
2005 {
2006 if (item.IsOk())
2007 [m_OutlineView selectRowIndexes:[NSIndexSet indexSetWithIndex:[m_OutlineView rowForItem:[m_DataSource getDataViewItemFromBuffer:item]]]
2008 byExtendingSelection:NO];
2009 }
2010
2011 void wxCocoaDataViewControl::SelectAll()
2012 {
2013 [m_OutlineView selectAll:m_OutlineView];
2014 }
2015
2016 void wxCocoaDataViewControl::Unselect(const wxDataViewItem& item)
2017 {
2018 if (item.IsOk())
2019 [m_OutlineView deselectRow:[m_OutlineView rowForItem:[m_DataSource getDataViewItemFromBuffer:item]]];
2020 }
2021
2022 void wxCocoaDataViewControl::UnselectAll()
2023 {
2024 [m_OutlineView deselectAll:m_OutlineView];
2025 }
2026
2027 //
2028 // sorting related methods
2029 //
2030 wxDataViewColumn* wxCocoaDataViewControl::GetSortingColumn() const
2031 {
2032 NSArray* const columns = [m_OutlineView tableColumns];
2033
2034 UInt32 const noOfColumns = [columns count];
2035
2036
2037 for (UInt32 i=0; i<noOfColumns; ++i)
2038 if ([[columns objectAtIndex:i] sortDescriptorPrototype] != nil)
2039 return static_cast<wxDataViewColumn*>([[[columns objectAtIndex:i] identifier] pointer]);
2040 return NULL;
2041 }
2042
2043 void wxCocoaDataViewControl::Resort()
2044 {
2045 [m_DataSource clearChildren];
2046 [m_OutlineView reloadData];
2047 }
2048
2049 //
2050 // other methods (inherited from wxDataViewWidgetImpl)
2051 //
2052 void wxCocoaDataViewControl::DoSetIndent(int indent)
2053 {
2054 [m_OutlineView setIndentationPerLevel:static_cast<CGFloat>(indent)];
2055 }
2056
2057 void wxCocoaDataViewControl::HitTest(const wxPoint& point, wxDataViewItem& item, wxDataViewColumn*& columnPtr) const
2058 {
2059 NSPoint const nativePoint = wxToNSPoint((NSScrollView*) GetWXWidget(),point);
2060
2061 int indexColumn;
2062 int indexRow;
2063
2064
2065 indexColumn = [m_OutlineView columnAtPoint:nativePoint];
2066 indexRow = [m_OutlineView rowAtPoint: nativePoint];
2067 if ((indexColumn >= 0) && (indexRow >= 0))
2068 {
2069 columnPtr = static_cast<wxDataViewColumn*>([[[[m_OutlineView tableColumns] objectAtIndex:indexColumn] identifier] pointer]);
2070 item = wxDataViewItem([[m_OutlineView itemAtRow:indexRow] pointer]);
2071 }
2072 else
2073 {
2074 columnPtr = NULL;
2075 item = wxDataViewItem();
2076 }
2077 }
2078
2079 void wxCocoaDataViewControl::SetRowHeight(const wxDataViewItem& WXUNUSED(item), unsigned int WXUNUSED(height))
2080 // Not supported by the native control
2081 {
2082 }
2083
2084 void wxCocoaDataViewControl::OnSize()
2085 {
2086 if ([m_OutlineView numberOfColumns] == 1)
2087 [m_OutlineView sizeLastColumnToFit];
2088 }
2089
2090 //
2091 // drag & drop helper methods
2092 //
2093 wxDataFormat wxCocoaDataViewControl::GetDnDDataFormat(wxDataObjectComposite* dataObjects)
2094 {
2095 wxDataFormat resultFormat;
2096 if ( !dataObjects )
2097 return resultFormat;
2098
2099 bool compatible(true);
2100
2101 size_t const noOfFormats = dataObjects->GetFormatCount();
2102 size_t indexFormat;
2103
2104 wxDataFormat* formats;
2105
2106 // get all formats and check afterwards if the formats are compatible; if
2107 // they are compatible the preferred format is returned otherwise
2108 // wxDF_INVALID is returned;
2109 // currently compatible types (ordered by priority are):
2110 // - wxDF_UNICODETEXT - wxDF_TEXT
2111 formats = new wxDataFormat[noOfFormats];
2112 dataObjects->GetAllFormats(formats);
2113 indexFormat = 0;
2114 while ((indexFormat < noOfFormats) && compatible)
2115 {
2116 switch (resultFormat.GetType())
2117 {
2118 case wxDF_INVALID:
2119 resultFormat.SetType(formats[indexFormat].GetType()); // first format (should only be reached if indexFormat == 0)
2120 break;
2121 case wxDF_TEXT:
2122 if (formats[indexFormat].GetType() == wxDF_UNICODETEXT)
2123 resultFormat.SetType(wxDF_UNICODETEXT);
2124 else // incompatible
2125 {
2126 resultFormat.SetType(wxDF_INVALID);
2127 compatible = false;
2128 }
2129 break;
2130 case wxDF_UNICODETEXT:
2131 if (formats[indexFormat].GetType() != wxDF_TEXT)
2132 {
2133 resultFormat.SetType(wxDF_INVALID);
2134 compatible = false;
2135 }
2136 break;
2137 default:
2138 resultFormat.SetType(wxDF_INVALID); // not (yet) supported format
2139 compatible = false;
2140 }
2141 ++indexFormat;
2142 }
2143
2144 delete[] formats;
2145
2146 return resultFormat;
2147 }
2148
2149 wxDataObjectComposite* wxCocoaDataViewControl::GetDnDDataObjects(NSData* dataObject) const
2150 {
2151 wxDataFormatId dataFormatID;
2152
2153
2154 [dataObject getBytes:&dataFormatID length:sizeof(wxDataFormatId)];
2155 switch (dataFormatID)
2156 {
2157 case wxDF_TEXT:
2158 case wxDF_UNICODETEXT:
2159 {
2160 wxTextDataObject* textDataObject(new wxTextDataObject());
2161
2162 if (textDataObject->SetData(wxDataFormat(dataFormatID),[dataObject length]-sizeof(wxDataFormatId),static_cast<char const*>([dataObject bytes])+sizeof(wxDataFormatId)))
2163 {
2164 wxDataObjectComposite* dataObjectComposite(new wxDataObjectComposite());
2165
2166 dataObjectComposite->Add(textDataObject);
2167 return dataObjectComposite;
2168 }
2169 else
2170 {
2171 delete textDataObject;
2172 return NULL;
2173 }
2174 }
2175 break;
2176 default:
2177 return NULL;
2178 }
2179 }
2180
2181 // ----------------------------------------------------------------------------
2182 // wxDataViewRendererNativeData
2183 // ----------------------------------------------------------------------------
2184
2185 void wxDataViewRendererNativeData::Init()
2186 {
2187 m_origFont = NULL;
2188 m_origTextColour = NULL;
2189 m_ellipsizeMode = wxELLIPSIZE_MIDDLE;
2190
2191 if ( m_ColumnCell )
2192 ApplyLineBreakMode(m_ColumnCell);
2193 }
2194
2195 void wxDataViewRendererNativeData::ApplyLineBreakMode(NSCell *cell)
2196 {
2197 NSLineBreakMode nsMode = NSLineBreakByWordWrapping;
2198 switch ( m_ellipsizeMode )
2199 {
2200 case wxELLIPSIZE_NONE:
2201 nsMode = NSLineBreakByClipping;
2202 break;
2203
2204 case wxELLIPSIZE_START:
2205 nsMode = NSLineBreakByTruncatingHead;
2206 break;
2207
2208 case wxELLIPSIZE_MIDDLE:
2209 nsMode = NSLineBreakByTruncatingMiddle;
2210 break;
2211
2212 case wxELLIPSIZE_END:
2213 nsMode = NSLineBreakByTruncatingTail;
2214 break;
2215 }
2216
2217 wxASSERT_MSG( nsMode != NSLineBreakByWordWrapping, "unknown wxEllipsizeMode" );
2218
2219 [cell setLineBreakMode: nsMode];
2220 }
2221
2222 // ---------------------------------------------------------
2223 // wxDataViewRenderer
2224 // ---------------------------------------------------------
2225
2226 wxDataViewRenderer::wxDataViewRenderer(const wxString& varianttype,
2227 wxDataViewCellMode mode,
2228 int align)
2229 : wxDataViewRendererBase(varianttype, mode, align),
2230 m_alignment(align),
2231 m_mode(mode),
2232 m_NativeDataPtr(NULL)
2233 {
2234 }
2235
2236 wxDataViewRenderer::~wxDataViewRenderer()
2237 {
2238 delete m_NativeDataPtr;
2239 }
2240
2241 void wxDataViewRenderer::SetAlignment(int align)
2242 {
2243 m_alignment = align;
2244 [GetNativeData()->GetColumnCell() setAlignment:ConvertToNativeHorizontalTextAlignment(align)];
2245 }
2246
2247 void wxDataViewRenderer::SetMode(wxDataViewCellMode mode)
2248 {
2249 m_mode = mode;
2250 if ( GetOwner() )
2251 [GetOwner()->GetNativeData()->GetNativeColumnPtr() setEditable:(mode == wxDATAVIEW_CELL_EDITABLE)];
2252 }
2253
2254 void wxDataViewRenderer::SetNativeData(wxDataViewRendererNativeData* newNativeDataPtr)
2255 {
2256 delete m_NativeDataPtr;
2257 m_NativeDataPtr = newNativeDataPtr;
2258 }
2259
2260 void wxDataViewRenderer::EnableEllipsize(wxEllipsizeMode mode)
2261 {
2262 // we need to store this value to apply it to the columns headerCell in
2263 // CreateNativeColumn()
2264 GetNativeData()->SetEllipsizeMode(mode);
2265
2266 // but we may already apply it to the column cell which will be used for
2267 // this column
2268 GetNativeData()->ApplyLineBreakMode(GetNativeData()->GetColumnCell());
2269 }
2270
2271 wxEllipsizeMode wxDataViewRenderer::GetEllipsizeMode() const
2272 {
2273 return GetNativeData()->GetEllipsizeMode();
2274 }
2275
2276 void
2277 wxDataViewRenderer::OSXOnCellChanged(NSObject *object,
2278 const wxDataViewItem& item,
2279 unsigned col)
2280 {
2281 // TODO: we probably should get rid of this code entirely and make this
2282 // function pure virtual, but currently we still have some native
2283 // renderers (wxDataViewChoiceRenderer) which don't override it and
2284 // there is also wxDataViewCustomRenderer for which it's not obvious
2285 // how it should be implemented so keep this "auto-deduction" of
2286 // variant type from NSObject for now
2287
2288 wxVariant value;
2289 if ( [object isKindOfClass:[NSString class]] )
2290 value = ObjectToString(object);
2291 else if ( [object isKindOfClass:[NSNumber class]] )
2292 value = ObjectToLong(object);
2293 else if ( [object isKindOfClass:[NSDate class]] )
2294 value = ObjectToDate(object);
2295 else
2296 {
2297 wxFAIL_MSG( wxString::Format
2298 (
2299 "unknown value type %s",
2300 wxCFStringRef::AsString([object className])
2301 ));
2302 return;
2303 }
2304
2305 wxDataViewModel *model = GetOwner()->GetOwner()->GetModel();
2306 model->ChangeValue(value, item, col);
2307 }
2308
2309 IMPLEMENT_ABSTRACT_CLASS(wxDataViewRenderer,wxDataViewRendererBase)
2310
2311 // ---------------------------------------------------------
2312 // wxDataViewCustomRenderer
2313 // ---------------------------------------------------------
2314 wxDataViewCustomRenderer::wxDataViewCustomRenderer(const wxString& varianttype,
2315 wxDataViewCellMode mode,
2316 int align)
2317 : wxDataViewRenderer(varianttype, mode, align),
2318 m_editorCtrlPtr(NULL),
2319 m_DCPtr(NULL)
2320 {
2321 SetNativeData(new wxDataViewRendererNativeData([[wxCustomCell alloc] init]));
2322 }
2323
2324 bool wxDataViewCustomRenderer::MacRender()
2325 {
2326 [GetNativeData()->GetItemCell() setObjectValue:[[[wxCustomRendererObject alloc] initWithRenderer:this] autorelease]];
2327 return true;
2328 }
2329
2330 IMPLEMENT_ABSTRACT_CLASS(wxDataViewCustomRenderer, wxDataViewRenderer)
2331
2332 // ---------------------------------------------------------
2333 // wxDataViewTextRenderer
2334 // ---------------------------------------------------------
2335 wxDataViewTextRenderer::wxDataViewTextRenderer(const wxString& varianttype,
2336 wxDataViewCellMode mode,
2337 int align)
2338 : wxDataViewRenderer(varianttype,mode,align)
2339 {
2340 NSTextFieldCell* cell;
2341
2342
2343 cell = [[NSTextFieldCell alloc] init];
2344 [cell setAlignment:ConvertToNativeHorizontalTextAlignment(align)];
2345 SetNativeData(new wxDataViewRendererNativeData(cell));
2346 [cell release];
2347 }
2348
2349 bool wxDataViewTextRenderer::MacRender()
2350 {
2351 if (GetValue().GetType() == GetVariantType())
2352 {
2353 [GetNativeData()->GetItemCell() setObjectValue:wxCFStringRef(GetValue().GetString()).AsNSString()];
2354 return true;
2355 }
2356 else
2357 {
2358 wxFAIL_MSG(wxString("Text renderer cannot render value because of wrong value type; value type: ") << GetValue().GetType());
2359 return false;
2360 }
2361 }
2362
2363 void
2364 wxDataViewTextRenderer::OSXOnCellChanged(NSObject *value,
2365 const wxDataViewItem& item,
2366 unsigned col)
2367 {
2368 wxDataViewModel *model = GetOwner()->GetOwner()->GetModel();
2369 model->ChangeValue(ObjectToString(value), item, col);
2370 }
2371
2372 IMPLEMENT_CLASS(wxDataViewTextRenderer,wxDataViewRenderer)
2373
2374 // ---------------------------------------------------------
2375 // wxDataViewBitmapRenderer
2376 // ---------------------------------------------------------
2377 wxDataViewBitmapRenderer::wxDataViewBitmapRenderer(const wxString& varianttype,
2378 wxDataViewCellMode mode,
2379 int align)
2380 : wxDataViewRenderer(varianttype,mode,align)
2381 {
2382 NSImageCell* cell;
2383
2384
2385 cell = [[NSImageCell alloc] init];
2386 SetNativeData(new wxDataViewRendererNativeData(cell));
2387 [cell release];
2388 }
2389
2390 bool wxDataViewBitmapRenderer::MacRender()
2391 // This method returns 'true' if
2392 // - the passed bitmap is valid and it could be assigned to the native data browser;
2393 // - the passed bitmap is invalid (or is not initialized); this case simulates a non-existing bitmap.
2394 // In all other cases the method returns 'false'.
2395 {
2396 wxCHECK_MSG(GetValue().GetType() == GetVariantType(),false,wxString("Bitmap renderer cannot render value; value type: ") << GetValue().GetType());
2397
2398 wxBitmap bitmap;
2399
2400 bitmap << GetValue();
2401 if (bitmap.IsOk())
2402 [GetNativeData()->GetItemCell() setObjectValue:[[bitmap.GetNSImage() retain] autorelease]];
2403 return true;
2404 }
2405
2406 IMPLEMENT_CLASS(wxDataViewBitmapRenderer,wxDataViewRenderer)
2407
2408 // -------------------------------------
2409 // wxDataViewChoiceRenderer
2410 // -------------------------------------
2411 wxDataViewChoiceRenderer::wxDataViewChoiceRenderer(const wxArrayString& choices,
2412 wxDataViewCellMode mode,
2413 int alignment)
2414 : wxDataViewRenderer(wxT("string"),mode,alignment), m_Choices(choices)
2415 {
2416 NSPopUpButtonCell* cell;
2417
2418
2419 cell = [[NSPopUpButtonCell alloc] init];
2420 [cell setControlSize:NSMiniControlSize];
2421 [cell setFont:[[NSFont fontWithName:[[cell font] fontName] size:[NSFont systemFontSizeForControlSize:NSMiniControlSize]] autorelease]];
2422 for (size_t i=0; i<choices.GetCount(); ++i)
2423 [cell addItemWithTitle:[[wxCFStringRef(choices[i]).AsNSString() retain] autorelease]];
2424 SetNativeData(new wxDataViewRendererNativeData(cell));
2425 [cell release];
2426 }
2427
2428 bool wxDataViewChoiceRenderer::MacRender()
2429 {
2430 if (GetValue().GetType() == GetVariantType())
2431 {
2432 [((NSPopUpButtonCell*) GetNativeData()->GetItemCell()) selectItemWithTitle:[[wxCFStringRef(GetValue().GetString()).AsNSString() retain] autorelease]];
2433 return true;
2434 }
2435 else
2436 {
2437 wxFAIL_MSG(wxString("Choice renderer cannot render value because of wrong value type; value type: ") << GetValue().GetType());
2438 return false;
2439 }
2440 }
2441
2442 IMPLEMENT_CLASS(wxDataViewChoiceRenderer,wxDataViewRenderer)
2443
2444 // ---------------------------------------------------------
2445 // wxDataViewDateRenderer
2446 // ---------------------------------------------------------
2447
2448 wxDataViewDateRenderer::wxDataViewDateRenderer(const wxString& varianttype,
2449 wxDataViewCellMode mode,
2450 int align)
2451 : wxDataViewRenderer(varianttype,mode,align)
2452 {
2453 NSTextFieldCell* cell;
2454
2455 NSDateFormatter* dateFormatter;
2456
2457
2458 dateFormatter = [[NSDateFormatter alloc] init];
2459 [dateFormatter setFormatterBehavior:NSDateFormatterBehavior10_4];
2460 [dateFormatter setDateStyle:NSDateFormatterShortStyle];
2461 cell = [[NSTextFieldCell alloc] init];
2462 [cell setFormatter:dateFormatter];
2463 SetNativeData(new wxDataViewRendererNativeData(cell,[NSDate dateWithString:@"2000-12-30 20:00:00 +0000"]));
2464 [cell release];
2465 [dateFormatter release];
2466 }
2467
2468 bool wxDataViewDateRenderer::MacRender()
2469 {
2470 if (GetValue().GetType() == GetVariantType())
2471 {
2472 if (GetValue().GetDateTime().IsValid())
2473 {
2474 // -- find best fitting style to show the date --
2475 // as the style should be identical for all cells a reference date instead of the actual cell's date
2476 // value is used for all cells; this reference date is stored in the renderer's native data section
2477 // for speed purposes; otherwise, the reference date's string has to be recalculated for each item that
2478 // may become timewise long if a lot of rows using dates exist;
2479 // the algorithm has the preference to display as much information as possible in the first instance;
2480 // but as this is often impossible due to space restrictions the style is shortened per loop; finally,
2481 // if the shortest time and date format does not fit into the cell the time part is dropped;
2482 // remark: the time part itself is not modified per iteration loop and only uses the short style,
2483 // means that only the hours and minutes are being shown
2484 [GetNativeData()->GetItemCell() setObjectValue:GetNativeData()->GetObject()]; // GetObject() returns a date for testing the size of a date object
2485 [[GetNativeData()->GetItemCell() formatter] setTimeStyle:NSDateFormatterShortStyle];
2486 for (int dateFormatterStyle=4; dateFormatterStyle>0; --dateFormatterStyle)
2487 {
2488 [[GetNativeData()->GetItemCell() formatter] setDateStyle:(NSDateFormatterStyle)dateFormatterStyle];
2489 if (dateFormatterStyle == 1)
2490 {
2491 // if the shortest style for displaying the date and time is too long to be fully visible remove the time part of the date:
2492 if ([GetNativeData()->GetItemCell() cellSize].width > [GetNativeData()->GetColumnPtr() width])
2493 [[GetNativeData()->GetItemCell() formatter] setTimeStyle:NSDateFormatterNoStyle];
2494 break; // basically not necessary as the loop would end anyway but let's save the last comparison
2495 }
2496 else if ([GetNativeData()->GetItemCell() cellSize].width <= [GetNativeData()->GetColumnPtr() width])
2497 break;
2498 }
2499 // set data (the style is set by the previous loop);
2500 // on OSX the date has to be specified with respect to UTC; in wxWidgets the date is always entered in the local timezone; so, we have to do a conversion
2501 // from the local to UTC timezone when adding the seconds to 1970-01-01 UTC:
2502 [GetNativeData()->GetItemCell() setObjectValue:[NSDate dateWithTimeIntervalSince1970:GetValue().GetDateTime().ToUTC().Subtract(wxDateTime(1,wxDateTime::Jan,1970)).GetSeconds().ToDouble()]];
2503 }
2504 return true;
2505 }
2506 else
2507 {
2508 wxFAIL_MSG(wxString("Date renderer cannot render value because of wrong value type; value type: ") << GetValue().GetType());
2509 return false;
2510 }
2511 }
2512
2513 void
2514 wxDataViewDateRenderer::OSXOnCellChanged(NSObject *value,
2515 const wxDataViewItem& item,
2516 unsigned col)
2517 {
2518 wxDataViewModel *model = GetOwner()->GetOwner()->GetModel();
2519 model->ChangeValue(ObjectToDate(value), item, col);
2520 }
2521
2522 IMPLEMENT_ABSTRACT_CLASS(wxDataViewDateRenderer,wxDataViewRenderer)
2523
2524 // ---------------------------------------------------------
2525 // wxDataViewIconTextRenderer
2526 // ---------------------------------------------------------
2527 wxDataViewIconTextRenderer::wxDataViewIconTextRenderer(const wxString& varianttype,
2528 wxDataViewCellMode mode,
2529 int align)
2530 : wxDataViewRenderer(varianttype,mode)
2531 {
2532 wxImageTextCell* cell;
2533
2534
2535 cell = [[wxImageTextCell alloc] init];
2536 [cell setAlignment:ConvertToNativeHorizontalTextAlignment(align)];
2537 SetNativeData(new wxDataViewRendererNativeData(cell));
2538 [cell release];
2539 }
2540
2541 bool wxDataViewIconTextRenderer::MacRender()
2542 {
2543 if (GetValue().GetType() == GetVariantType())
2544 {
2545 wxDataViewIconText iconText;
2546
2547 wxImageTextCell* cell;
2548
2549 cell = (wxImageTextCell*) GetNativeData()->GetItemCell();
2550 iconText << GetValue();
2551 if (iconText.GetIcon().IsOk())
2552 [cell setImage:[[wxBitmap(iconText.GetIcon()).GetNSImage() retain] autorelease]];
2553 [cell setStringValue:[[wxCFStringRef(iconText.GetText()).AsNSString() retain] autorelease]];
2554 return true;
2555 }
2556 else
2557 {
2558 wxFAIL_MSG(wxString("Icon & text renderer cannot render value because of wrong value type; value type: ") << GetValue().GetType());
2559 return false;
2560 }
2561 }
2562
2563 void
2564 wxDataViewIconTextRenderer::OSXOnCellChanged(NSObject *value,
2565 const wxDataViewItem& item,
2566 unsigned col)
2567 {
2568 wxVariant valueIconText;
2569 valueIconText << wxDataViewIconText(ObjectToString(value));
2570
2571 wxDataViewModel *model = GetOwner()->GetOwner()->GetModel();
2572 model->ChangeValue(valueIconText, item, col);
2573 }
2574
2575 IMPLEMENT_ABSTRACT_CLASS(wxDataViewIconTextRenderer,wxDataViewRenderer)
2576
2577 // ---------------------------------------------------------
2578 // wxDataViewToggleRenderer
2579 // ---------------------------------------------------------
2580 wxDataViewToggleRenderer::wxDataViewToggleRenderer(const wxString& varianttype,
2581 wxDataViewCellMode mode,
2582 int align)
2583 : wxDataViewRenderer(varianttype,mode)
2584 {
2585 NSButtonCell* cell;
2586
2587
2588 cell = [[NSButtonCell alloc] init];
2589 [cell setAlignment:ConvertToNativeHorizontalTextAlignment(align)];
2590 [cell setButtonType:NSSwitchButton];
2591 [cell setImagePosition:NSImageOnly];
2592 SetNativeData(new wxDataViewRendererNativeData(cell));
2593 [cell release];
2594 }
2595
2596 bool wxDataViewToggleRenderer::MacRender()
2597 {
2598 if (GetValue().GetType() == GetVariantType())
2599 {
2600 [GetNativeData()->GetItemCell() setIntValue:GetValue().GetLong()];
2601 return true;
2602 }
2603 else
2604 {
2605 wxFAIL_MSG(wxString("Toggle renderer cannot render value because of wrong value type; value type: ") << GetValue().GetType());
2606 return false;
2607 }
2608 }
2609
2610 void
2611 wxDataViewToggleRenderer::OSXOnCellChanged(NSObject *value,
2612 const wxDataViewItem& item,
2613 unsigned col)
2614 {
2615 wxDataViewModel *model = GetOwner()->GetOwner()->GetModel();
2616 model->ChangeValue(ObjectToBool(value), item, col);
2617 }
2618
2619 IMPLEMENT_ABSTRACT_CLASS(wxDataViewToggleRenderer,wxDataViewRenderer)
2620
2621 // ---------------------------------------------------------
2622 // wxDataViewProgressRenderer
2623 // ---------------------------------------------------------
2624 wxDataViewProgressRenderer::wxDataViewProgressRenderer(const wxString& label,
2625 const wxString& varianttype,
2626 wxDataViewCellMode mode,
2627 int align)
2628 : wxDataViewRenderer(varianttype,mode,align)
2629 {
2630 NSLevelIndicatorCell* cell;
2631
2632 cell = [[NSLevelIndicatorCell alloc] initWithLevelIndicatorStyle:NSContinuousCapacityLevelIndicatorStyle];
2633 [cell setMinValue:0];
2634 [cell setMaxValue:100];
2635 SetNativeData(new wxDataViewRendererNativeData(cell));
2636 [cell release];
2637 }
2638
2639 bool wxDataViewProgressRenderer::MacRender()
2640 {
2641 if (GetValue().GetType() == GetVariantType())
2642 {
2643 [GetNativeData()->GetItemCell() setIntValue:GetValue().GetLong()];
2644 return true;
2645 }
2646 else
2647 {
2648 wxFAIL_MSG(wxString("Progress renderer cannot render value because of wrong value type; value type: ") << GetValue().GetType());
2649 return false;
2650 }
2651 }
2652
2653 void
2654 wxDataViewProgressRenderer::OSXOnCellChanged(NSObject *value,
2655 const wxDataViewItem& item,
2656 unsigned col)
2657 {
2658 wxDataViewModel *model = GetOwner()->GetOwner()->GetModel();
2659 model->ChangeValue(ObjectToLong(value), item, col);
2660 }
2661
2662 IMPLEMENT_ABSTRACT_CLASS(wxDataViewProgressRenderer,wxDataViewRenderer)
2663
2664 // ---------------------------------------------------------
2665 // wxDataViewColumn
2666 // ---------------------------------------------------------
2667
2668 wxDataViewColumn::wxDataViewColumn(const wxString& title,
2669 wxDataViewRenderer* renderer,
2670 unsigned int model_column,
2671 int width,
2672 wxAlignment align,
2673 int flags)
2674 : wxDataViewColumnBase(renderer, model_column),
2675 m_NativeDataPtr(new wxDataViewColumnNativeData()),
2676 m_title(title)
2677 {
2678 InitCommon(width, align, flags);
2679 if (renderer && (renderer->GetAlignment() == wxDVR_DEFAULT_ALIGNMENT))
2680 renderer->SetAlignment(align);
2681 }
2682
2683 wxDataViewColumn::wxDataViewColumn(const wxBitmap& bitmap,
2684 wxDataViewRenderer* renderer,
2685 unsigned int model_column,
2686 int width,
2687 wxAlignment align,
2688 int flags)
2689 : wxDataViewColumnBase(bitmap, renderer, model_column),
2690 m_NativeDataPtr(new wxDataViewColumnNativeData())
2691 {
2692 InitCommon(width, align, flags);
2693 if (renderer && (renderer->GetAlignment() == wxDVR_DEFAULT_ALIGNMENT))
2694 renderer->SetAlignment(align);
2695 }
2696
2697 wxDataViewColumn::~wxDataViewColumn()
2698 {
2699 delete m_NativeDataPtr;
2700 }
2701
2702 bool wxDataViewColumn::IsSortKey() const
2703 {
2704 NSTableColumn *nsCol = GetNativeData()->GetNativeColumnPtr();
2705 return nsCol && ([nsCol sortDescriptorPrototype] != nil);
2706 }
2707
2708 void wxDataViewColumn::SetAlignment(wxAlignment align)
2709 {
2710 m_alignment = align;
2711 [[m_NativeDataPtr->GetNativeColumnPtr() headerCell] setAlignment:ConvertToNativeHorizontalTextAlignment(align)];
2712 if (m_renderer && (m_renderer->GetAlignment() == wxDVR_DEFAULT_ALIGNMENT))
2713 m_renderer->SetAlignment(align);
2714 }
2715
2716 void wxDataViewColumn::SetBitmap(const wxBitmap& bitmap)
2717 {
2718 // bitmaps and titles cannot exist at the same time - if the bitmap is set the title is removed:
2719 m_title = wxEmptyString;
2720 wxDataViewColumnBase::SetBitmap(bitmap);
2721 [[m_NativeDataPtr->GetNativeColumnPtr() headerCell] setImage:[[bitmap.GetNSImage() retain] autorelease]];
2722 }
2723
2724 void wxDataViewColumn::SetMaxWidth(int maxWidth)
2725 {
2726 m_maxWidth = maxWidth;
2727 [m_NativeDataPtr->GetNativeColumnPtr() setMaxWidth:maxWidth];
2728 }
2729
2730 void wxDataViewColumn::SetMinWidth(int minWidth)
2731 {
2732 m_minWidth = minWidth;
2733 [m_NativeDataPtr->GetNativeColumnPtr() setMinWidth:minWidth];
2734 }
2735
2736 void wxDataViewColumn::SetReorderable(bool reorderable)
2737 {
2738 }
2739
2740 void wxDataViewColumn::SetResizeable(bool resizeable)
2741 {
2742 wxDataViewColumnBase::SetResizeable(resizeable);
2743 if (resizeable)
2744 [m_NativeDataPtr->GetNativeColumnPtr() setResizingMask:NSTableColumnUserResizingMask];
2745 else
2746 [m_NativeDataPtr->GetNativeColumnPtr() setResizingMask:NSTableColumnNoResizing];
2747 }
2748
2749 void wxDataViewColumn::SetSortable(bool sortable)
2750 {
2751 wxDataViewColumnBase::SetSortable(sortable);
2752 }
2753
2754 void wxDataViewColumn::SetSortOrder(bool ascending)
2755 {
2756 if (m_ascending != ascending)
2757 {
2758 m_ascending = ascending;
2759 if (IsSortKey())
2760 {
2761 // change sorting order:
2762 NSArray* sortDescriptors;
2763 NSSortDescriptor* sortDescriptor;
2764 NSTableColumn* tableColumn;
2765
2766 tableColumn = m_NativeDataPtr->GetNativeColumnPtr();
2767 sortDescriptor = [[NSSortDescriptor alloc] initWithKey:[[tableColumn sortDescriptorPrototype] key] ascending:m_ascending];
2768 sortDescriptors = [NSArray arrayWithObject:sortDescriptor];
2769 [tableColumn setSortDescriptorPrototype:sortDescriptor];
2770 [[tableColumn tableView] setSortDescriptors:sortDescriptors];
2771 [sortDescriptor release];
2772 }
2773 }
2774 }
2775
2776 void wxDataViewColumn::SetTitle(const wxString& title)
2777 {
2778 // bitmaps and titles cannot exist at the same time - if the title is set the bitmap is removed:
2779 wxDataViewColumnBase::SetBitmap(wxBitmap());
2780 m_title = title;
2781 [[m_NativeDataPtr->GetNativeColumnPtr() headerCell] setStringValue:[[wxCFStringRef(title).AsNSString() retain] autorelease]];
2782 }
2783
2784 void wxDataViewColumn::SetWidth(int width)
2785 {
2786 [m_NativeDataPtr->GetNativeColumnPtr() setWidth:width];
2787 m_width = width;
2788 }
2789
2790 void wxDataViewColumn::SetAsSortKey(bool WXUNUSED(sort))
2791 {
2792 // see wxGTK native wxDataViewColumn implementation
2793 wxFAIL_MSG("not implemented");
2794 }
2795
2796 void wxDataViewColumn::SetNativeData(wxDataViewColumnNativeData* newNativeDataPtr)
2797 {
2798 delete m_NativeDataPtr;
2799 m_NativeDataPtr = newNativeDataPtr;
2800 }
2801
2802 #endif // (wxUSE_DATAVIEWCTRL == 1) && !defined(wxUSE_GENERICDATAVIEWCTRL)