5 static void UnusedFunction()
14 a.OnEdit(null,null,0,0,0,0,0);
15 a.OnDisplay(null,0,0,0,0,0,0);
19 extern int __ecereVMethodID_class_OnEdit;
20 extern int __ecereVMethodID_class_OnDisplay;
21 extern int __ecereVMethodID_class_OnGetString;
22 extern int __ecereVMethodID_class_OnFree;
23 extern int __ecereVMethodID_class_OnCompare;
24 extern int __ecereVMethodID_class_OnCopy;
25 extern int __ecereVMethodID_class_OnSaveEdit;
27 //#define SHOW_METHODS
30 // *** THESE METHODS SHOULD BE IMPROVED UPON AND USED TO SET PROPERTIES FROM NOW ON ***
31 // (Other locations where this will be useful: JSON (DataValue version?), CodeEditor, ...)
32 // This takes in a value according to the any_object rules
34 void SetPropValue(Property prop, void * object, any_object value)
36 Class type = prop.dataTypeClass;
38 type = prop.dataTypeClass = eSystem_FindClass(prop._class.module, prop.dataTypeString);
40 if(type.type == normalClass || type.type == noHeadClass || type.type == structClass)
42 prop.Set(object, value);
44 // TOFIX: How to swiftly handle classes with base data type?
45 else if(type == class(double) || !strcmp(type.dataTypeString, "double"))
47 ((void (*)(void *, double))(void *)prop.Set)(object, *(double *)value);
49 else if(type == class(float) || !strcmp(type.dataTypeString, "float"))
51 ((void (*)(void *, float))(void *)prop.Set)(object, *(float *)value);
53 else if(type.typeSize == sizeof(int64))// || !strcmp(type.dataTypeString, "int64") || !strcmp(type.dataTypeString, "unsigned int64") || !strcmp(type.dataTypeString, "uint64"))
55 ((void (*)(void *, int64))(void *)prop.Set)(object, *(int64 *)value);
57 else if(type.typeSize == sizeof(int))// || !strcmp(type.dataTypeString, "int") || !strcmp(type.dataTypeString, "unsigned int") || !strcmp(type.dataTypeString, "uint"))
59 ((void (*)(void *, int))(void *)prop.Set)(object, *(int *)value);
61 else if(type.typeSize == sizeof(short int)) // || !strcmp(type.dataTypeString, "short") || !strcmp(type.dataTypeString, "unsigned short") || !strcmp(type.dataTypeString, "uint16") || !strcmp(type.dataTypeString, "int16"))
63 ((void (*)(void *, short))(void *)prop.Set)(object, *(short *)value);
65 else if(type.typeSize == sizeof(byte))// || !strcmp(type.dataTypeString, "char") || !strcmp(type.dataTypeString, "unsigned char") || !strcmp(type.dataTypeString, "byte"))
67 ((void (*)(void *, byte))(void *)prop.Set)(object, *(byte *)value);
71 ((void (*)(void *, void *))(void *)prop.Set)(object, value);
75 any_object GetPropValue(Property prop, Instance object)
79 Class type = prop.dataTypeClass;
82 type = prop.dataTypeClass = eSystem_FindClass(prop._class.module, prop.dataTypeString);
85 if(type.type == normalClass || type.type == noHeadClass || type.type == structClass)
87 return ((void*(*)())(void *)prop.Get)(object);
89 // TOFIX: How to swiftly handle classes with base data type?
90 else if(type == class(double) || !strcmp(type.dataTypeString, "double"))
92 double d = ((double(*)())(void *)prop.Get)(object);
95 else if(type == class(float) || !strcmp(type.dataTypeString, "float"))
97 float f =((float(*)())(void *)prop.Get)(object);
100 else if(type.typeSize == sizeof(int64))// || !strcmp(type.dataTypeString, "int64") || !strcmp(type.dataTypeString, "unsigned int64") || !strcmp(type.dataTypeString, "uint64"))
102 return ((int64(*)())(void *)prop.Get)(object);
104 else if(type.typeSize == sizeof(int))// || !strcmp(type.dataTypeString, "int") || !strcmp(type.dataTypeString, "unsigned int") || !strcmp(type.dataTypeString, "uint"))
106 return ((int(*)())(void *)prop.Get)(object);
108 else if(type.typeSize == sizeof(short int)) // || !strcmp(type.dataTypeString, "short") || !strcmp(type.dataTypeString, "unsigned short") || !strcmp(type.dataTypeString, "uint16") || !strcmp(type.dataTypeString, "int16"))
110 return ((short(*)())(void *)prop.Get)(object);
112 else if(type.typeSize == sizeof(byte))// || !strcmp(type.dataTypeString, "char") || !strcmp(type.dataTypeString, "unsigned char") || !strcmp(type.dataTypeString, "byte"))
114 return ((byte(*)())(void *)prop.Get)(object);
118 return prop.Get(object);
125 void CopyProperty(Property prop, Instance dest, Instance src)
127 Class type = prop.dataTypeClass;
129 type = prop.dataTypeClass = eSystem_FindClass(prop._class.module, prop.dataTypeString);
131 if(type.type == structClass)
133 void * propData = new0 byte[type.structSize];
134 prop.Get(src, propData);
135 prop.Set(dest, propData);
138 else if(type.type == normalClass || type.type == noHeadClass)
140 // TOCHECK: Why was there a return here?
141 /*return */prop.Set(dest, ((void*(*)())(void *)prop.Get)(src));
143 // TOFIX: How to swiftly handle classes with base data type?
144 else if(type == class(double) || !strcmp(type.dataTypeString, "double"))
146 ((void (*)(void *, double))(void *)prop.Set)(dest, ((double(*)())(void *)prop.Get)(src));
148 else if(type == class(float) || !strcmp(type.dataTypeString, "float"))
150 ((void (*)(void *, float))(void *)prop.Set)(dest, ((float(*)())(void *)prop.Get)(src));
152 else if(type.typeSize == sizeof(int64))// || !strcmp(type.dataTypeString, "int64") || !strcmp(type.dataTypeString, "unsigned int64") || !strcmp(type.dataTypeString, "uint64"))
154 ((void (*)(void *, int64))(void *)prop.Set)(dest, ((int64(*)())(void *)prop.Get)(src));
156 else if(type.typeSize == sizeof(int))// || !strcmp(type.dataTypeString, "int") || !strcmp(type.dataTypeString, "unsigned int") || !strcmp(type.dataTypeString, "uint"))
158 ((void (*)(void *, int))(void *)prop.Set)(dest, ((int(*)())(void *)prop.Get)(src));
160 else if(type.typeSize == sizeof(short int)) // || !strcmp(type.dataTypeString, "short") || !strcmp(type.dataTypeString, "unsigned short") || !strcmp(type.dataTypeString, "uint16") || !strcmp(type.dataTypeString, "int16"))
162 ((void (*)(void *, short))(void *)prop.Set)(dest, ((short(*)())(void *)prop.Get)(src));
164 else if(type.typeSize == sizeof(byte))// || !strcmp(type.dataTypeString, "char") || !strcmp(type.dataTypeString, "unsigned char") || !strcmp(type.dataTypeString, "byte"))
166 ((void (*)(void *, byte))(void *)prop.Set)(dest, ((byte(*)())(void *)prop.Get)(src));
170 prop.Set(dest, prop.Get(src));
174 void GetProperty(Property prop, Instance object, DataValue value)
178 Class type = prop.dataTypeClass;
181 type = prop.dataTypeClass = eSystem_FindClass(prop._class.module, prop.dataTypeString);
183 if(prop._class.module.application == __thisModule &&
184 prop.dataTypeClass.module.application == ((Designer)GetActiveDesigner()).codeEditor.privateModule)
189 if(type.type == normalClass || type.type == noHeadClass || type.type == structClass)
191 value.p = ((void*(*)())(void *)prop.Get)(object);
193 // TOFIX: How to swiftly handle classes with base data type?
194 else if(type == class(double) || !strcmp(type.dataTypeString, "double"))
196 value.d = ((double(*)())(void *)prop.Get)(object);
198 else if(type == class(float) || !strcmp(type.dataTypeString, "float"))
200 value.f = ((float(*)())(void *)prop.Get)(object);
202 else if(type.typeSize == sizeof(int64))// || !strcmp(type.dataTypeString, "int64") || !strcmp(type.dataTypeString, "unsigned int64") || !strcmp(type.dataTypeString, "uint64"))
204 value.i64 = ((int64(*)())(void *)prop.Get)(object);
206 else if(type.typeSize == sizeof(int))// || !strcmp(type.dataTypeString, "int") || !strcmp(type.dataTypeString, "unsigned int") || !strcmp(type.dataTypeString, "uint"))
208 value.i = ((int(*)())(void *)prop.Get)(object);
210 else if(type.typeSize == sizeof(short int)) // || !strcmp(type.dataTypeString, "short") || !strcmp(type.dataTypeString, "unsigned short") || !strcmp(type.dataTypeString, "uint16") || !strcmp(type.dataTypeString, "int16"))
212 value.s = ((short(*)())(void *)prop.Get)(object);
214 else if(type.typeSize == sizeof(byte))// || !strcmp(type.dataTypeString, "char") || !strcmp(type.dataTypeString, "unsigned char") || !strcmp(type.dataTypeString, "byte"))
216 value.uc = ((byte(*)())(void *)prop.Get)(object);
220 value.i = prop.Get(object);
227 void SetProperty(Property prop, Instance object, DataValue value)
231 Class type = prop.dataTypeClass;
233 type = prop.dataTypeClass = eSystem_FindClass(prop._class.module, prop.dataTypeString);
235 if(type.type == normalClass || type.type == noHeadClass || type.type == structClass)
237 prop.Set(object, value);
239 // TOFIX: How to swiftly handle classes with base data type?
240 else if(type == class(double) || !strcmp(type.dataTypeString, "double"))
242 ((void (*)(void *, double))(void *)prop.Set)(object, value.d);
244 else if(type == class(float) || !strcmp(type.dataTypeString, "float"))
246 ((void (*)(void *, float))(void *)prop.Set)(object, value.f);
248 else if(type.typeSize == sizeof(int64))// || !strcmp(type.dataTypeString, "int64") || !strcmp(type.dataTypeString, "unsigned int64") || !strcmp(type.dataTypeString, "uint64"))
250 ((void (*)(void *, int64))(void *)prop.Set)(object, value.i64);
252 else if(type.typeSize == sizeof(int))// || !strcmp(type.dataTypeString, "int") || !strcmp(type.dataTypeString, "unsigned int") || !strcmp(type.dataTypeString, "uint"))
254 ((void (*)(void *, int))(void *)prop.Set)(object, value.i);
256 else if(type.typeSize == sizeof(short int)) // || !strcmp(type.dataTypeString, "short") || !strcmp(type.dataTypeString, "unsigned short") || !strcmp(type.dataTypeString, "uint16") || !strcmp(type.dataTypeString, "int16"))
258 ((void (*)(void *, short))(void *)prop.Set)(object, value.s);
260 else if(type.typeSize == sizeof(byte))// || !strcmp(type.dataTypeString, "char") || !strcmp(type.dataTypeString, "unsigned char") || !strcmp(type.dataTypeString, "byte"))
262 ((void (*)(void *, byte))(void *)prop.Set)(object, value.uc);
266 ((void (*)(void *, int))(void *)prop.Set)(object, value.i);
274 borderStyle = sizable;
278 anchor = { left = 0, top = 0, bottom = 0 };
279 background = activeBorder;
283 dropBox.AddField(dropField);
284 properties.AddField(propertyName);
285 properties.AddField(propertyValue);
286 methods.AddField(methodName);
291 categories.Free(null);
297 anchor = { left = 0, top = 0, right = 0 };
299 bool NotifySelect(DropBox control, DataRow row, Modifiers keyFlags)
301 ObjectInfo selected = (ObjectInfo)(row ? row.tag : null);
302 ToolBox toolBox = ((IDE)parent).toolBox;
304 if(codeEditor && selected)
305 codeEditor.SelectObject(selected);
307 // TODO: Get containing class of object
308 toolBox.selectedClass = selected ? selected.oClass : null;
310 object = selected ? selected.instance : null;
313 ListProperties(true);
317 row = methods.currentRow;
319 strcpy(selectedMethod, ((CodeObject)row.GetData(methodName)).name);
323 if(selected && selected.instance && codeEditor)
327 int rowHeight = methods.rowHeight;
329 propertyValue.userData = (void *)selected.instance;
331 // Fill up the methods
335 for(_class = selected.instance._class; _class && _class.type != systemClass; _class = _class.base)
338 for(id = _class.base ? _class.base.vTblSize : 0; id<_class.vTblSize; id++)
341 for(method = (Method)_class.methods.first; method; method = (Method)((BTNode)method).next)
343 if(method.type == virtualMethod && method.vid == id)
346 method.dataType = ProcessTypeString(method.dataTypeString, false);
348 type = method.dataType.thisClass ? typeEvent : typeMethod;
350 DataRow row = methods.AddRow();
351 CodeObject codeObject
353 eventsUp = (selected.oClass == selected) ? false : true;
357 overriden = codeEditor.FindMethod(method.name, &codeObject.function, null);
359 if(!codeObject.overriden || codeObject.overriden == 2)
360 codeEditor.FindCompatibleMethods(method, codeObject.compatible);
362 row.SetData(methodName, codeObject);
364 if(!strcmp(method.name, selectedMethod))
365 methods.currentRow = row;
373 methods.Sort(methodName, 1);
376 for(row = methods.firstRow; row; row = row.next)
378 CodeObject codeObject = row.GetData(methodName);
379 CreateButtons(codeObject, row.index * rowHeight, rowHeight, row);
386 DataField dropField { dataType = class(CodeObject) };
390 this, inactive = true, text = $"Properties", bevelOver = true, isRadio = true;
392 anchor = { left = 0, bottom = 3, right = 0.5 };
395 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
397 text = $"Properties";
398 button.font = { "Tahoma", 8.25f, bold = true };
401 methods.visible = false;
402 methBtn.Activate(); // Ensure proper cycling (until tab order?)
403 properties.visible = true;
405 alphabetical.disabled = false;
406 categorized.disabled = false;
408 properties.Activate();
410 // ((IDE)master).SheetSelected(Properties);
417 this, inactive = true, bevelOver = true;
422 anchor = { bottom = 3, left = 0.5, right = 0 };
424 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
427 button.font = { "Tahoma", 8.25f, bold = true };
430 properties.visible = false;
431 methBtn.Activate(); // Ensure proper cycling (until tab order?)
432 methods.visible = true;
434 alphabetical.disabled = true;
435 categorized.disabled = true;
439 // ((IDE)master).SheetSelected(Methods);
446 MenuPlacement editMenu { menu, text = $"Edit" };
447 Menu viewMenu { menu, text = $"View" };
452 this, anchor = { left = 0, right = 0, top = 50, bottom = 25 };
453 hasVertScroll = true, alwaysEdit = true, collapseControl = true, resizable = true;
454 background = viewsBackground;
455 foreground = viewsText;
457 bool NotifySelect(ListBox control, DataRow row, Modifiers keyFlags)
462 strcpy(selectedProp, (char *)row.GetData(propertyName));
463 selectedScroll = properties.scroll.y;
464 selectedScroll -= row.index * properties.rowHeight;
470 prop = ((PropertyInfo)row.GetData(propertyValue)).prop;
479 DataField propertyName { dataType = class(char *), width = 130 };
480 DataField propertyValue { dataType = class(PropertyInfo), width = 0, editable = true };
485 this, anchor = { left = 0, right = 0, top = 50, bottom = 25 };
486 hasVertScroll = true;
487 background = viewsBackground;
488 foreground = viewsText;
489 // alwaysEdit = true;
492 bool NotifyDoubleClick(ListBox control, int x, int y, Modifiers mods)
494 CodeObject object = control.GetData(methodName);
496 codeEditor.AddMethod(object.method);
500 bool NotifyRightClick(ListBox control, int x, int y, Modifiers mods)
502 CodeObject object = control.GetData(methodName);
506 if(object.overriden == 0)
508 MenuItem { menu, $"Override", o, enter, bold = true, NotifySelect = OverrideMethodSelected };
509 if(object.compatible.count)
511 Menu attachMenu { menu, $"Attach", a };
513 for(compatible = object.compatible.first; compatible; compatible = compatible.next)
515 ClassFunction function = compatible.data;
516 MenuItem { attachMenu, function.declarator.symbol.string, id = (int)function, NotifySelect = AttachMethodSelected };
520 else if(object.overriden == 1)
522 MenuItem { menu, $"Go to", g, enter, bold = true, NotifySelect = GotoMethodSelected };
523 MenuItem { menu, $"Detach", d, d, NotifySelect = DetachMethodSelected };
524 MenuItem { menu, $"Delete", del, del, NotifySelect = DeleteMethodSelected };
526 else if(object.overriden == 2)
528 MenuItem { menu, $"Go to", g, enter, bold = true, NotifySelect = GotoMethodSelected };
529 MenuItem { menu, $"Detach", d, d, NotifySelect = DetachMethodSelected };
530 if(object.compatible.count > 1)
532 Menu attachMenu { menu, $"Reattach", r };
534 for(compatible = object.compatible.first; compatible; compatible = compatible.next)
536 ClassFunction function = compatible.data;
537 if(function != object.function)
539 MenuItem { attachMenu, function.declarator.symbol.string, id = (int)function, NotifySelect = ReattachMethodSelected };
545 attachMethod = object.method;
546 popupMenu = PopupMenu { menu = menu, master = this, position =
548 x + control.absPosition.x - app.desktop.absPosition.x,
549 y + control.absPosition.y - app.desktop.absPosition.y
552 // popupMenu.Capture();
556 bool NotifyKeyDown(ListBox listBox, DataRow row, Key key, unichar ch)
560 CodeObject object = row.GetData(methodName);
561 switch((SmartKey)key)
564 if(!object.overriden)
565 listBox.NotifyDoubleClick(this, listBox, 0,0, 0);
567 codeEditor.GoToMethod(object.method.name);
571 object.deleteBtn.NotifyClicked(this, object.deleteBtn, 0,0,0);
572 else if(object.detachBtn)
573 object.detachBtn.NotifyClicked(this, object.detachBtn, 0,0,0);
577 object.attachBtn.NotifyPushed(this, object.attachBtn, 0,0,0);
581 object.detachBtn.NotifyClicked(this, object.detachBtn, 0,0,0);
589 DataField methodName { dataType = class(CodeObject) };
592 methBtn.font = { "Tahoma", 8.25, bold = true };
593 methBtn.checked = true;
594 properties.visible = false;
597 propBtn.font = { "Tahoma", 8.25f, bold = true };
598 propBtn.checked = true;
599 methods.visible = false;
600 text = $"Properties";
605 this, bevelOver = true, inactive = true, position = { 25, 25 }, size = { 24, 24 };
606 bitmap = { "<:ecere>elements/orderAscending.png" };
609 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
611 if(!alphabetical.checked)
613 alphabetical.checked = true;
614 categorized.checked = false;
616 ListProperties(true);
624 this, bevelOver = true, checked = true, inactive = true, position = { 0, 25 }, size = { 24, 24 };
625 bitmap = { "<:ecere>elements/orderCategorized.png" };
628 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
630 if(!categorized.checked)
632 categorized.checked = true;
633 alphabetical.checked = false;
635 ListProperties(true);
641 property CodeEditor codeEditor
645 if(codeEditor != value)
650 dropField.userData = codeEditor;
651 methodName.userData = codeEditor;
654 codeEditor.EnumerateObjects(this);
663 property SheetType sheetSelected
667 if(methBtn.checked != (value == SheetType::methods))
672 return methBtn.checked ? methods : properties;
676 bool OnClose(bool parentClosing)
686 bool OnKeyDown(Key key, unichar ch)
692 else if(key == escape)
694 Window activeClient = ide.activeClient;
696 activeClient.Activate();
698 ide.RepositionWindows(true);
703 bool OnKeyHit(Key key, unichar ch)
705 return properties.OnKeyHit(key, ch);
708 bool OnActivate(bool active, Window previous, bool * goOnWithActivation, bool direct)
710 if(active && codeEditor)
711 codeEditor.EnsureUpToDate();
715 void ListProperties(bool clear)
717 DataRow row = dropBox.currentRow;
718 ObjectInfo selected = row ? (ObjectInfo)row.tag : null;
721 bool categorized = this.categorized.checked;
722 bool currentRow = false;
723 char selectedProp[1024];
728 DataRow row = properties.currentRow;
731 DataRow propRow = row;
733 while(propRow && propRow.parent && !propRow.parent.isHeader)
734 propRow = row.parent;
736 propName = propRow.GetData(propertyName);
737 strcpy(this.selectedProp, propName);
738 selectedScroll = properties.scroll.y;
739 selectedScroll -= propRow.index * properties.rowHeight;
741 currentRow = this.selectedProp[0] ? true : false;
743 for(cat = categories.first; cat; cat = cat.next)
745 cat.collapsed = cat.row.collapsed;
749 // Preserve selected property (PropertySheetSelect will null it)
750 strcpy(selectedProp, this.selectedProp);
752 strcpy(this.selectedProp, selectedProp);
754 if(selected && selected.instance && codeEditor)
756 Instance test = eInstance_New(selected.instance._class);
760 // Put it in the same desktop window...
761 if(selected.classDefinition)
762 codeEditor.designer.PrepareTestObject(test);
764 // Fill up the properties
765 while(_class != selected.instance._class)
767 BitMember bitMember = null;
768 Class lastClass = _class;
771 for(_class = selected.instance._class; _class.base != lastClass && _class.base.type != systemClass && _class.inheritanceAccess != privateAccess; _class = _class.base);
773 for(propIt = _class.membersAndProperties.first; propIt; propIt = propIt.next)
775 if(propIt.isProperty)
777 Property prop = eClass_FindProperty(selected.instance._class, propIt.name, GetPrivateModule());
779 if(prop && prop.Set && prop.Get && prop.compiled)
781 bool disabled = Code_IsPropertyDisabled(selected, prop.name);
783 Class dataType = prop.dataTypeClass;
785 dataType = prop.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, prop.dataTypeString);
787 if(!strcmp(_class.name, "DesignerBase"))
789 bold = !disabled && Code_IsPropertyModified(test, selected, prop);
794 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont };
795 char * name = prop.category ? prop.category : $"Misc";
796 Category category = categories.FindName(name, false);
798 // Hide properties like this for now..
799 if(name && !strcmp(name, "Private"))
804 category = Category { name = name };
805 categories.AddName(category);
807 if(!category.row && categorized)
809 PropertyInfo catInfo { null, false, null, name };
810 category.row = properties.AddRow();
811 category.row.SetData(propertyName, name );
812 category.row.SetData(propertyValue, catInfo);
813 category.row.isHeader = true;
814 category.row.collapsed = category.collapsed;
819 row = categorized ? category.row.FindRow((int)prop) : properties.FindRow((int)prop);
821 row = categorized ? category.row.AddRow() : properties.AddRow();
825 row = categorized ? category.row.FindRow((int)prop) : properties.FindRow((int)prop);
827 row.SetData(propertyName, prop.name);
828 row.SetData(propertyValue, info);
830 if(clear && !strcmp(prop.name, this.selectedProp))
831 properties.currentRow = row;
833 if(!dataType.noExpansion && (dataType.type == structClass || dataType.type == normalClass || dataType.type == noHeadClass || dataType.type == bitClass))
838 row.collapsed = true;
840 for(member = dataType.membersAndProperties.first; member; member = member.next)
842 if(member.isProperty)
844 Property subProp = (Property)member;
845 if(!subProp.conversion && subProp.Get && subProp.Set)
848 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont, null, null, subProp };
852 subRow = row.AddRow();
853 subRow.tag = (int)subProp;
856 subRow = row.FindRow((int)subProp);
858 subRow.SetData(propertyName, subProp.name);
859 subRow.SetData(propertyValue, info);
865 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont, null, member, null };
868 subRow = row.AddRow();
869 subRow.tag = (int)member;
872 subRow = row.FindRow((int)member);
874 subRow.SetData(propertyName, member.name);
875 subRow.SetData(propertyValue, info);
879 DataMember subMember;
880 for(subMember = member.members.first; subMember; subMember = subMember.next)
883 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont, null, subMember, null, member.offset };
886 subRow = row.AddRow();
887 subRow.tag = (int)subMember;
890 subRow = row.FindRow((int)subMember);
892 subRow.SetData(propertyName, subMember.name);
893 subRow.SetData(propertyValue, info);
904 // Sort alphabetically for now...
907 // properties.Sort(null, 1);
908 properties.Sort(propertyValue, 1);
909 if(!properties.currentRow)
913 for(_class = selected.instance._class; _class; _class = _class.base)
916 for(prop = _class.membersAndProperties.first; prop; prop = prop.next)
918 if(prop.isProperty && prop.Set && prop.Get && prop.compiled)
920 if(_class.defaultProperty && !strcmp(prop.name, _class.defaultProperty))
923 char * name = prop.category ? prop.category : $"Misc";
924 Category category = categories.FindName(name, false);
925 row = category ? (categorized ? category.row.FindRow((int)prop) : properties.FindRow((int)prop)) : null;
926 properties.currentRow = row;
935 properties.currentRow = properties.firstRow;
940 DataRow row = properties.currentRow;
941 properties.scroll.y = selectedScroll + row.index * properties.rowHeight;
947 void AddObject(ObjectInfo object, char * name, CodeObjectType type, bool select)
949 DataRow after = null;
951 CodeObject codeObject;
952 char * bitmap = null;
953 bool foundClass = false;
955 for(row = dropBox.firstRow; row; row = row.next)
957 CodeObject codeObject = row.GetData(null);
958 if(codeObject.object.oClass == object.oClass)
965 row = (DataRow)dropBox.AddRowAfter(after);
967 row.tag = (int)object;
974 indent = (type == typeClass) ? 0 : 1;
977 if(type != typeClass)
978 bitmap = (char *)eClass_GetProperty(object.instance._class, "icon");
981 codeObject.bitmap = { bitmap };
982 AddResource(codeObject.bitmap);
985 row.SetData(null, codeObject);
989 this.object = object ? object.instance : null;
990 propertyValue.userData = object ? (void *)object.instance : null;
991 dropBox.SelectRow(row);
995 void DeleteObject(ObjectInfo object)
997 DataRow row = dropBox.FindRow((int)object);
1000 CodeObject codeObject = row.GetData(null);
1002 if(codeObject.bitmap)
1003 RemoveResource(codeObject.bitmap);
1004 dropBox.DeleteRow(row);
1008 void SelectObject(ObjectInfo object)
1012 DataRow row = dropBox.FindRow((int)object);
1013 this.object = object ? object.instance : null;
1014 propertyValue.userData = object ? (void *)object.instance : null;
1015 dropBox.SelectRow(row);
1019 void RenameObject(ObjectInfo object, char * name)
1021 DataRow row = dropBox.FindRow((int)object);
1022 CodeObject codeObject = row.GetData(null);
1023 // Isn't this useless? Shouldn't it be after?
1024 codeObject.name = name;
1025 // row.SetData(null, codeObject); // Is this necessary?
1028 void DataBox::EditSetData(any_object setValue, bool closingDropDown)
1030 ((Sheet)master.master).SetData(setValue, this);
1033 void SetData(any_object setValue, DataBox dataBox)
1035 //PropertyInfo propertyPtr = row.GetData(null);
1036 PropertyInfo propertyPtr = properties.GetData(null);
1037 Property prop = propertyPtr ? propertyPtr.prop : null;
1038 Instance object = this.object;
1041 Class dataType = prop.dataTypeClass;
1043 dataType = prop.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, prop.dataTypeString);
1044 if(propertyPtr.subMember)
1046 DataMember member = propertyPtr.subMember;
1047 Class subDataType = member.dataTypeClass;
1048 if(!member.dataTypeClass)
1049 subDataType = member.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, member.dataTypeString);
1053 if(!subDataType.dataType)
1054 subDataType.dataType = ProcessTypeString(subDataType.dataTypeString, false);
1056 if(dataType.type == structClass)
1058 data = new0 byte[dataType.structSize];
1059 prop.Get(object, data);
1060 // CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, &setValue, subDataType.size);
1061 CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, (void *)setValue, subDataType.dataType.size);
1062 prop.Set(object, data);
1064 else if(dataType.type == normalClass || dataType.type == noHeadClass)
1069 if(dataType.type == bitClass)
1071 BitMember bitMember = (BitMember) member;
1074 DataValue value = { 0 };
1075 value.ui = prop.Get(object);
1076 value.ui &= ~ (uint)bitMember.mask;
1077 value.ui |= *(uint32 *)setValue << bitMember.pos;
1078 prop.Set(object, value.ui);
1083 data = dataType.typeSize ? new0 byte[dataType.typeSize] : null;
1084 prop.Get(object, data);
1085 // CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, &setValue, subDataType.typeSize);
1086 CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, (void *)setValue, subDataType.dataType.size);
1087 // TODO: Support non 32 bit datatypes here
1088 prop.Set(object, data);
1092 if(data) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType,&data);
1096 else if(propertyPtr.subProperty)
1098 Property subProperty = propertyPtr.subProperty;
1099 Class subDataType = subProperty.dataTypeClass;
1101 subDataType = subProperty.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, subProperty.dataTypeString);
1106 if(dataType.type == structClass)
1108 data = new0 byte[dataType.structSize];
1109 prop.Get(object, data);
1110 subProperty.Set(data, *(uint32 *)setValue);
1111 prop.Set(object, data);
1113 else if(dataType.type == normalClass || dataType.type == noHeadClass)
1115 Instance current = (Instance)prop.Get(object);
1116 Instance propObject = eInstance_New(dataType);
1117 CopyInstanceData(dataType, propObject, current);
1118 subProperty.Set(propObject, (uint32)setValue);
1119 prop.Set(object, propObject);
1123 data = dataType.typeSize ? new0 byte[dataType.typeSize] : null;
1124 prop.Get(object, data);
1125 subProperty.Set(data, (uint32)setValue);
1126 // TODO: Support not 32 bit data types here
1127 prop.Set(object, data);
1130 if(data) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType,&data);
1136 SetPropValue(prop, object, (uint32)setValue);
1138 Code_FixProperty(propertyPtr.prop, object);
1140 properties.Update(null);
1141 dropBox.Update(null);
1142 codeEditor.designer.Update(null);
1143 codeEditor.Update(null); // patch for redraw bug if on top
1145 ListProperties(false);
1147 dataBox.editor.font = { propertyPtr.font.faceName, propertyPtr.font.size, propertyPtr.font.bold };
1148 codeEditor.ModifyCode();
1152 bool SaveEdit(PropertyInfo propertyPtr, Instance object)
1154 codeEditor.designer.Update(null);
1155 codeEditor.Update(null); // patch for redraw bug if on top
1156 properties.Update(null);
1157 dropBox.Update(null);
1159 Code_FixProperty(propertyPtr.prop, object);
1160 ListProperties(false);
1162 codeEditor.ModifyCode();
1168 if(!propBtn.checked)
1170 propBtn.checked = true;
1171 propBtn.NotifyClicked(this, propBtn, 0,0,0);
1175 methBtn.checked = true;
1176 methBtn.NotifyClicked(this, methBtn, 0,0,0);
1180 bool AttachMethodSelected(MenuItem selection, Modifiers mods)
1182 ClassFunction function = (ClassFunction)selection.id;
1183 codeEditor.AttachMethod(attachMethod, function);
1187 bool ReattachMethodSelected(MenuItem selection, Modifiers mods)
1189 ClassFunction function = (ClassFunction)selection.id;
1190 CodeObject object = methods.GetData(methodName);
1191 codeEditor.ReAttachMethod(attachMethod, function);
1195 bool OverrideMethodSelected(MenuItem selection, Modifiers mods)
1197 ClassFunction function = (ClassFunction)selection.id;
1198 CodeObject object = methods.GetData(methodName);
1200 codeEditor.AddMethod(object.method);
1204 bool GotoMethodSelected(MenuItem selection, Modifiers mods)
1206 ClassFunction function = (ClassFunction)selection.id;
1207 CodeObject object = methods.GetData(methodName);
1209 codeEditor.GoToMethod(object.method.name);
1213 bool DetachMethodSelected(MenuItem selection, Modifiers mods)
1215 ClassFunction function = (ClassFunction)selection.id;
1216 CodeObject object = methods.GetData(methodName);
1218 codeEditor.DetachMethod(object.method, object.function, object.overriden);
1222 bool DeleteMethodSelected(MenuItem selection, Modifiers mods)
1224 ClassFunction function = (ClassFunction)selection.id;
1225 CodeObject object = methods.GetData(methodName);
1227 object.deleteBtn.NotifyClicked(this, object.deleteBtn, 0,0,0);
1231 bool AddMethodClicked(Button button, int x, int y, Modifiers mods)
1233 DataRow row = (DataRow)button.id;
1234 CodeObject object = row.GetData(methodName);
1235 codeEditor.AddMethod(object.method);
1239 void CreateButtons(CodeObject codeObject, int y, int h, DataRow row)
1241 BitmapResource bitmap;
1243 if(codeObject.overriden)
1245 if(codeObject.overriden == 1)
1247 codeObject.deleteBtn = Button
1249 methods, master = this,
1251 bitmap = { ":actions/delete.png", alphaBlend = true },
1252 anchor = { right = 16, top = y },
1256 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
1258 CodeObject codeObject = ((DataRow)button.id).GetData(null);
1259 bool confirmation = !Code_IsFunctionEmpty(codeObject.function, codeObject.method, codeEditor.selected);
1264 sprintf(title, $"Delete %s", codeObject.name);
1267 master = parent, type = okCancel, text = title,
1268 contents = $"Method still contains code. Are you sure you want to delete it?"
1270 confirmation = false;
1273 if(!confirmation && codeObject.function.attached.count)
1276 sprintf(title, $"Delete %s", codeObject.name);
1277 confirmation = true;
1280 master = parent, type = okCancel, text = title,
1281 contents = $"Other methods are still attached to this method. Are you sure you want to delete it?"
1283 confirmation = false;
1288 codeEditor.DeleteMethod(codeObject.function);
1293 incref codeObject.deleteBtn;
1294 codeObject.deleteBtn.Create();
1297 if(codeObject.overriden == 2 || !codeObject.function.attached.count)
1299 codeObject.detachBtn = Button
1302 master = methods.master,
1304 bitmap = { ":actions/detach.png" },
1305 anchor = { right = 0, top = y },
1309 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
1311 DataRow row = (DataRow)button.id;
1312 CodeObject object = row.GetData(methodName);
1314 codeEditor.DetachMethod(object.method, object.function, object.overriden);
1318 incref codeObject.detachBtn;
1319 codeObject.detachBtn.Create();
1324 if(codeObject.compatible.count)
1326 codeObject.attachBtn = Button
1328 parent = methods, master = methods.master,
1330 bitmap = { ":actions/attach.png" },
1331 anchor = { right = 0, top = y },
1335 bool NotifyPushed(Button button, int x, int y, Modifiers mods)
1338 DataRow row = (DataRow)button.id;
1339 CodeObject object = row.GetData(methodName);
1341 PopupMenu popupMenu;
1345 for(compatible = object.compatible.first; compatible; compatible = compatible.next)
1347 ClassFunction function = compatible.data;
1348 MenuItem { menu, function.declarator.symbol.string, id = (int)function, NotifySelect = AttachMethodSelected };
1350 attachMethod = object.method;
1352 popupMenu = PopupMenu
1354 master = this, menu = menu,
1357 button.absPosition.x - app.desktop.position.x,
1358 button.absPosition.y - app.desktop.position.y + button.size.h
1362 button.ReleaseCapture();
1363 popupMenu.Capture();
1367 incref codeObject.attachBtn;
1368 codeObject.attachBtn.Create();
1373 Method attachMethod;
1374 char selectedMethod[1024];
1375 CodeEditor codeEditor;
1377 char selectedProp[1024];
1381 static int String_OnCompare(char ** string1, char ** string2)
1384 if(*string1 && *string2)
1385 result = strcmpi(*string1, *string2);
1386 else if(!*string1 && *string2)
1388 else if(*string1 && !*string2)
1393 static void CopyInstanceData(Class dataType, Instance propObject, Instance current)
1396 for(_class = dataType; _class && _class.type != systemClass; _class = _class.base)
1399 for(member = _class.membersAndProperties.first; member; member = member.next)
1401 Class memberType = member.dataTypeClass;
1403 memberType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1404 if(member.isProperty)
1406 Property subProp = (Property) member;
1407 if(subProp.Get && subProp.Set)
1408 CopyProperty(subProp, propObject, current);
1413 // TOCHECK: I have serious doubts this works in many cases.
1414 memberType._vTbl[__ecereVMethodID_class_OnCopy](memberType, (byte *)propObject + member.offset, (byte *)current + member.offset);
1415 else if(member.memberOffset)
1416 memcpy((byte *)propObject + member.offset, (byte *)current + member.offset, member.memberOffset);
1422 class PropertyInfo : struct
1428 char * categoryName;
1429 DataMember subMember;
1430 Property subProperty;
1433 void OnDisplay(Surface surface, int x, int y, int width, Instance object, Alignment alignment, DataDisplayFlags displayFlags)
1435 Property prop = this.prop;
1437 surface.TextFont(font.font);
1440 surface.SetBackground(Color { 170, 170, 170 });
1441 surface.Area(0,0, x+width-1, y+100);
1443 else if(prop && prop.dataTypeString)
1445 Class dataType = prop.dataTypeClass;
1446 Module module = ((Designer)GetActiveDesigner()).codeEditor.privateModule;
1448 dataType = prop.dataTypeClass = eSystem_FindClass(module, prop.dataTypeString);
1450 if(dataType && prop.Get)
1452 void * dataPtr, * data = null, * subData = null;
1453 DataValue valueData, valueSubData;
1457 if(dataType.type == structClass)
1459 data = new0 byte[dataType.structSize];
1460 prop.Get(object, data);
1465 GetProperty(prop, object, &valueData);
1467 if(dataType.type == normalClass)
1468 dataPtr = valueData.p;
1470 dataPtr = &valueData;
1476 DataMember member = this.subMember;
1477 Class subDataType = member.dataTypeClass;
1479 subDataType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1482 if(dataType.type == bitClass)
1484 BitMember bitMember = (BitMember)member;
1485 bitValue = (valueData.i & bitMember.mask) >> bitMember.pos;
1486 dataPtr = &bitValue;
1489 dataPtr = (byte *)dataPtr + member.offset + this.extraOffset;
1491 dataType = subDataType;
1493 else if(this.subProperty)
1495 Property subProperty = this.subProperty;
1496 Class subDataType = subProperty.dataTypeClass;
1498 subDataType = subProperty.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, subProperty.dataTypeString);
1499 if(!subProperty.Get) subDataType = null;
1502 if(subDataType.type == structClass)
1504 subData = new0 byte[subDataType.structSize];
1505 subProperty.Get(dataPtr, subData);
1510 GetProperty(subProperty, dataPtr, &valueSubData);
1511 if(subDataType.type == normalClass)
1512 dataPtr = valueSubData.p;
1514 dataPtr = &valueSubData;
1517 dataType = subDataType;
1521 dataType._vTbl[__ecereVMethodID_class_OnDisplay](dataType, dataPtr, surface, x, y, width, null, alignment, displayFlags);
1529 Window OnEdit(DataBox dataBox, Window obsolete, int x, int y, int w, int h, void * unused)
1531 EditBox editData = null;
1532 Property prop = this.prop;
1534 dataBox.SetData = Sheet::EditSetData;
1535 if(prop && prop.dataTypeString && !this.disabled)
1537 Sheet propertyWindow = (Sheet)dataBox.master.master;
1538 Instance object = propertyWindow.object;
1539 Class dataType = prop.dataTypeClass;
1541 dataType = prop.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, prop.dataTypeString);
1543 if(dataType && prop.Get)
1545 void * dataPtr, * data = null, * subData = null;
1546 DataValue valueData, valueSubData;
1550 if(dataType.type == structClass)
1552 data = new0 byte[dataType.structSize];
1553 prop.Get(object, data);
1558 GetProperty(prop, object, &valueData);
1559 if(dataType.type == normalClass)
1560 dataPtr = valueData.p;
1562 dataPtr = &valueData;
1568 DataMember member = this.subMember;
1569 Class subDataType = member.dataTypeClass;
1571 subDataType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1574 if(dataType.type == bitClass)
1576 BitMember bitMember = (BitMember)member;
1577 bitValue = (valueData.i & bitMember.mask) >> bitMember.pos;
1578 dataPtr = &bitValue;
1581 dataPtr = (byte *)dataPtr + member.offset + this.extraOffset;
1583 dataType = subDataType;
1585 else if(this.subProperty)
1587 Property subProperty = this.subProperty;
1588 Class subDataType = subProperty.dataTypeClass;
1590 subDataType = subProperty.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, subProperty.dataTypeString);
1591 if(!subProperty.Get) subDataType = null;
1594 if(subDataType.type == structClass)
1596 subData = new0 byte[subDataType.structSize];
1597 subProperty.Get(dataPtr, subData);
1602 GetProperty(subProperty, dataPtr, &valueSubData);
1603 if(subDataType.type == normalClass)
1604 dataPtr = valueSubData.p;
1606 dataPtr = &valueSubData;
1609 dataType = subDataType;
1613 editData = (void *)dataType._vTbl[__ecereVMethodID_class_OnEdit](dataType, dataPtr, dataBox, obsolete, x, y, w, h, object /*unused*/);
1618 editData.font = { font.faceName, font.size, font.bold };
1624 int OnCompare(PropertyInfo data2)
1626 char * category1 = prop ? prop.category : categoryName;
1627 char * category2 = data2.prop ? data2.prop.category : data2.categoryName;
1630 if(!category1) category1 = $"Misc";
1631 if(!category2) category2 = $"Misc";
1635 // result = String::OnCompare((String)category1, (String)category2);
1636 result = String_OnCompare(&category1, &category2);
1641 if(subMember && !data2.subMember)
1645 else if(!subMember && data2.subMember)
1649 else if(subMember && data2.subMember)
1651 if(subMember.id < data2.subMember.id)
1653 else if(subMember.id > data2.subMember.id)
1658 else if(subProperty && !data2.subProperty)
1662 else if(!subProperty && data2.subProperty)
1666 else if(subProperty && data2.subProperty)
1668 if(subProperty.id < data2.subProperty.id)
1670 else if(subProperty.id > data2.subProperty.id)
1675 else if(prop && !data2.prop)
1677 else if(!prop && data2.prop)
1680 // result = ((String)prop.name).OnCompare(data2.prop.name);
1681 // result = String::OnCompare((String)prop.name, (String)data2.prop.name);
1682 result = String_OnCompare(&prop.name, &data2.prop.name);
1687 bool OnSaveEdit(Window editControl, void * unusedData)
1689 Property prop = this.prop;
1692 Sheet sheet = (Sheet)editControl.master.master.master;
1693 Instance object = sheet.object;
1694 Class mainDataType = prop.dataTypeClass;
1696 bool result = false;
1697 void * dataPtr, * data = null, * subData = null;
1698 void * propObject = null;
1699 DataValue valueData = { 0 }, valueSubData = { 0 };
1703 mainDataType = prop.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, prop.dataTypeString);
1704 dataType = mainDataType;
1706 // Prepare main prop
1707 if(dataType.type == structClass)
1709 data = new0 byte[dataType.structSize];
1710 if(this.subMember || this.subProperty)
1711 prop.Get(object, data);
1715 else if(dataType.type == normalClass || dataType.type == noHeadClass)
1717 dataPtr = &valueData;
1719 if(this.subMember || this.subProperty)
1722 Instance current = (Instance)prop.Get(object);
1723 propObject = valueData.p = eInstance_New(dataType);
1724 CopyInstanceData(dataType, propObject, current);
1730 if(this.subMember || this.subProperty)
1731 GetProperty(prop, object, &valueData);
1733 dataPtr = &valueData;
1734 propObject = &valueData;
1740 DataMember member = this.subMember;
1741 Class subDataType = member.dataTypeClass;
1743 subDataType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1746 if(dataType.type == bitClass)
1747 dataPtr = &bitValue;
1749 dataPtr = (byte *)dataPtr + member.offset + this.extraOffset;
1751 dataType = subDataType;
1753 else if(this.subProperty)
1755 Property subProperty = this.subProperty;
1756 Class subDataType = subProperty.dataTypeClass;
1759 subDataType = subProperty.dataTypeClass =
1760 eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, subProperty.dataTypeString);
1761 if(!subProperty.Get) subDataType = null;
1764 if(subDataType.type == structClass)
1766 subData = new0 byte[subDataType.structSize];
1770 dataPtr = &valueSubData;
1772 dataType = subDataType;
1777 if(dataType._vTbl[__ecereVMethodID_class_OnSaveEdit](dataType, dataPtr, editControl, null))
1779 if(mainDataType.type == bitClass && this.subMember)
1781 BitMember bitMember = (BitMember)this.subMember;
1782 valueData.ui &= ~ (uint)bitMember.mask;
1783 valueData.ui |= bitValue << bitMember.pos;
1785 if(this.subProperty)
1787 if(dataType.type == structClass)
1788 this.subProperty.Set(propObject, subData);
1789 else if(dataType.type == unitClass || dataType.type == enumClass || dataType.type == bitClass)
1791 if(!strcmp(dataType.dataTypeString, "float"))
1792 ((void(*)(void *,float))(void *)this.subProperty.Set)(propObject, valueSubData.f);
1793 else if(!strcmp(dataType.dataTypeString, "double"))
1794 ((void(*)(void *,double))(void *)this.subProperty.Set)(propObject, valueSubData.d);
1795 else if(!strcmp(dataType.dataTypeString, "byte"))
1796 ((void(*)(void *,byte))(void *)this.subProperty.Set)(propObject, valueSubData.uc);
1797 else if(!strcmp(dataType.dataTypeString, "uint16"))
1798 ((void(*)(void *,uint16))(void *)this.subProperty.Set)(propObject, valueSubData.us);
1800 this.subProperty.Set(propObject, valueSubData.ui);
1803 this.subProperty.Set(propObject, valueSubData.ui);
1805 if(mainDataType.type == structClass)
1806 prop.Set(object, data);
1807 else if(mainDataType.type == unitClass || mainDataType.type == enumClass || mainDataType.type == bitClass)
1809 if(!strcmp(mainDataType.dataTypeString, "float"))
1810 ((void(*)(void *,float))(void *)prop.Set)(object, valueData.f);
1811 else if(!strcmp(mainDataType.dataTypeString, "double"))
1812 ((void(*)(void *,double))(void *)prop.Set)(object, valueData.d);
1813 else if(!strcmp(mainDataType.dataTypeString, "byte"))
1814 ((void(*)(void *,byte))(void *)prop.Set)(object, valueData.uc);
1815 else if(!strcmp(mainDataType.dataTypeString, "uint16"))
1816 ((void(*)(void *,uint16))(void *)prop.Set)(object, valueData.us);
1818 prop.Set(object, valueData.ui);
1821 prop.Set(object, valueData.ui);
1825 if(data == dataPtr) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType, &data);
1826 if(subData == dataPtr) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType, &subData);
1832 return sheet.SaveEdit(this, object);
1838 class Category : struct
1840 Category prev, next;