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)
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;
455 bool NotifySelect(ListBox control, DataRow row, Modifiers keyFlags)
460 strcpy(selectedProp, (char *)row.GetData(propertyName));
461 selectedScroll = properties.scroll.y;
462 selectedScroll -= row.index * properties.rowHeight;
468 prop = ((PropertyInfo)row.GetData(propertyValue)).prop;
477 DataField propertyName { dataType = class(char *), width = 130 };
478 DataField propertyValue { dataType = class(PropertyInfo), width = 0, editable = true };
483 this, anchor = { left = 0, right = 0, top = 50, bottom = 25 };
484 hasVertScroll = true;
485 // alwaysEdit = true;
488 bool NotifyDoubleClick(ListBox control, int x, int y, Modifiers mods)
490 CodeObject object = control.GetData(methodName);
492 codeEditor.AddMethod(object.method);
496 bool NotifyRightClick(ListBox control, int x, int y, Modifiers mods)
498 CodeObject object = control.GetData(methodName);
502 if(object.overriden == 0)
504 MenuItem { menu, "Override", o, enter, bold = true, NotifySelect = OverrideMethodSelected };
505 if(object.compatible.count)
507 Menu attachMenu { menu, "Attach", a };
509 for(compatible = object.compatible.first; compatible; compatible = compatible.next)
511 ClassFunction function = compatible.data;
512 MenuItem { attachMenu, function.declarator.symbol.string, id = (int)function, NotifySelect = AttachMethodSelected };
516 else if(object.overriden == 1)
518 MenuItem { menu, "Go to", g, enter, bold = true, NotifySelect = GotoMethodSelected };
519 MenuItem { menu, "Detach", d, d, NotifySelect = DetachMethodSelected };
520 MenuItem { menu, "Delete", del, del, NotifySelect = DeleteMethodSelected };
522 else if(object.overriden == 2)
524 MenuItem { menu, "Go to", g, enter, bold = true, NotifySelect = GotoMethodSelected };
525 MenuItem { menu, "Detach", d, d, NotifySelect = DetachMethodSelected };
526 if(object.compatible.count > 1)
528 Menu attachMenu { menu, "Reattach", r };
530 for(compatible = object.compatible.first; compatible; compatible = compatible.next)
532 ClassFunction function = compatible.data;
533 if(function != object.function)
535 MenuItem { attachMenu, function.declarator.symbol.string, id = (int)function, NotifySelect = ReattachMethodSelected };
541 attachMethod = object.method;
542 popupMenu = PopupMenu { menu = menu, master = this, position =
544 x + control.absPosition.x - app.desktop.absPosition.x,
545 y + control.absPosition.y - app.desktop.absPosition.y
548 // popupMenu.Capture();
552 bool NotifyKeyDown(ListBox listBox, DataRow row, Key key, unichar ch)
556 CodeObject object = row.GetData(methodName);
557 switch((SmartKey)key)
560 if(!object.overriden)
561 listBox.NotifyDoubleClick(this, listBox, 0,0, 0);
563 codeEditor.GoToMethod(object.method.name);
567 object.deleteBtn.NotifyClicked(this, object.deleteBtn, 0,0,0);
568 else if(object.detachBtn)
569 object.detachBtn.NotifyClicked(this, object.detachBtn, 0,0,0);
573 object.attachBtn.NotifyPushed(this, object.attachBtn, 0,0,0);
577 object.detachBtn.NotifyClicked(this, object.detachBtn, 0,0,0);
585 DataField methodName { dataType = class(CodeObject) };
588 methBtn.font = { "Tahoma", 8.25, bold = true };
589 methBtn.checked = true;
590 properties.visible = false;
593 propBtn.font = { "Tahoma", 8.25f, bold = true };
594 propBtn.checked = true;
595 methods.visible = false;
601 this, bevelOver = true, inactive = true, position = { 25, 25 }, size = { 24, 24 };
602 bitmap = { "<:ecere>elements/orderAscending.png" };
605 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
607 if(!alphabetical.checked)
609 alphabetical.checked = true;
610 categorized.checked = false;
612 ListProperties(true);
620 this, bevelOver = true, checked = true, inactive = true, position = { 0, 25 }, size = { 24, 24 };
621 bitmap = { "<:ecere>elements/orderCategorized.png" };
624 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
626 if(!categorized.checked)
628 categorized.checked = true;
629 alphabetical.checked = false;
631 ListProperties(true);
637 property CodeEditor codeEditor
641 if(codeEditor != value)
646 dropField.userData = codeEditor;
647 methodName.userData = codeEditor;
650 codeEditor.EnumerateObjects(this);
659 property SheetType sheetSelected
663 if(methBtn.checked != (value == SheetType::methods))
668 return methBtn.checked ? methods : properties;
672 bool OnClose(bool parentClosing)
682 bool OnKeyDown(Key key, unichar ch)
688 else if(key == escape)
690 Window activeClient = ide.activeClient;
692 activeClient.Activate();
694 ide.RepositionWindows(true);
699 bool OnKeyHit(Key key, unichar ch)
701 return properties.OnKeyHit(key, ch);
704 bool OnActivate(bool active, Window previous, bool * goOnWithActivation, bool direct)
706 if(active && codeEditor)
707 codeEditor.EnsureUpToDate();
711 void ListProperties(bool clear)
713 DataRow row = dropBox.currentRow;
714 ObjectInfo selected = row ? (ObjectInfo)row.tag : null;
717 bool categorized = this.categorized.checked;
718 bool currentRow = false;
719 char selectedProp[1024];
724 DataRow row = properties.currentRow;
727 DataRow propRow = row;
729 while(propRow && propRow.parent && !propRow.parent.isHeader)
730 propRow = row.parent;
732 propName = propRow.GetData(propertyName);
733 strcpy(this.selectedProp, propName);
734 selectedScroll = properties.scroll.y;
735 selectedScroll -= propRow.index * properties.rowHeight;
737 currentRow = this.selectedProp[0] ? true : false;
739 for(cat = categories.first; cat; cat = cat.next)
741 cat.collapsed = cat.row.collapsed;
745 // Preserve selected property (PropertySheetSelect will null it)
746 strcpy(selectedProp, this.selectedProp);
748 strcpy(this.selectedProp, selectedProp);
750 if(selected && selected.instance && codeEditor)
752 Instance test = eInstance_New(selected.instance._class);
756 // Put it in the same desktop window...
757 if(selected.classDefinition)
758 codeEditor.designer.PrepareTestObject(test);
760 // Fill up the properties
761 while(_class != selected.instance._class)
763 BitMember bitMember = null;
764 Class lastClass = _class;
767 for(_class = selected.instance._class; _class.base != lastClass && _class.base.type != systemClass && _class.inheritanceAccess != privateAccess; _class = _class.base);
769 for(propIt = _class.membersAndProperties.first; propIt; propIt = propIt.next)
771 if(propIt.isProperty)
773 Property prop = eClass_FindProperty(selected.instance._class, propIt.name, GetPrivateModule());
775 if(prop && prop.Set && prop.Get && prop.compiled)
777 bool disabled = Code_IsPropertyDisabled(selected, prop.name);
779 Class dataType = prop.dataTypeClass;
781 dataType = prop.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, prop.dataTypeString);
783 if(!strcmp(_class.name, "DesignerBase"))
785 bold = !disabled && Code_IsPropertyModified(test, selected, prop);
790 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont };
791 char * name = prop.category ? prop.category : "Misc";
792 Category category = categories.FindName(name, false);
794 // Hide properties like this for now..
795 if(name && !strcmp(name, "Private"))
800 category = Category { name = name };
801 categories.AddName(category);
803 if(!category.row && categorized)
805 PropertyInfo catInfo { null, false, null, name };
806 category.row = properties.AddRow();
807 category.row.SetData(propertyName, name );
808 category.row.SetData(propertyValue, catInfo);
809 category.row.isHeader = true;
810 category.row.collapsed = category.collapsed;
815 row = categorized ? category.row.FindRow((int)prop) : properties.FindRow((int)prop);
817 row = categorized ? category.row.AddRow() : properties.AddRow();
821 row = categorized ? category.row.FindRow((int)prop) : properties.FindRow((int)prop);
823 row.SetData(propertyName, prop.name);
824 row.SetData(propertyValue, info);
826 if(clear && !strcmp(prop.name, this.selectedProp))
827 properties.currentRow = row;
829 if(!dataType.noExpansion && (dataType.type == structClass || dataType.type == normalClass || dataType.type == noHeadClass || dataType.type == bitClass))
834 row.collapsed = true;
836 for(member = dataType.membersAndProperties.first; member; member = member.next)
838 if(member.isProperty)
840 Property subProp = (Property)member;
841 if(!subProp.conversion && subProp.Get && subProp.Set)
844 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont, null, null, subProp };
848 subRow = row.AddRow();
849 subRow.tag = (int)subProp;
852 subRow = row.FindRow((int)subProp);
854 subRow.SetData(propertyName, subProp.name);
855 subRow.SetData(propertyValue, info);
861 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont, null, member, null };
864 subRow = row.AddRow();
865 subRow.tag = (int)member;
868 subRow = row.FindRow((int)member);
870 subRow.SetData(propertyName, member.name);
871 subRow.SetData(propertyValue, info);
875 DataMember subMember;
876 for(subMember = member.members.first; subMember; subMember = subMember.next)
879 PropertyInfo info { prop, disabled, bold ? codeEditor.boldFont : codeEditor.normalFont, null, subMember, null, member.offset };
882 subRow = row.AddRow();
883 subRow.tag = (int)subMember;
886 subRow = row.FindRow((int)subMember);
888 subRow.SetData(propertyName, subMember.name);
889 subRow.SetData(propertyValue, info);
900 // Sort alphabetically for now...
903 // properties.Sort(null, 1);
904 properties.Sort(propertyValue, 1);
905 if(!properties.currentRow)
909 for(_class = selected.instance._class; _class; _class = _class.base)
912 for(prop = _class.membersAndProperties.first; prop; prop = prop.next)
914 if(prop.isProperty && prop.Set && prop.Get && prop.compiled)
916 if(_class.defaultProperty && !strcmp(prop.name, _class.defaultProperty))
919 char * name = prop.category ? prop.category : "Misc";
920 Category category = categories.FindName(name, false);
921 row = category ? (categorized ? category.row.FindRow((int)prop) : properties.FindRow((int)prop)) : null;
922 properties.currentRow = row;
931 properties.currentRow = properties.firstRow;
936 DataRow row = properties.currentRow;
937 properties.scroll.y = selectedScroll + row.index * properties.rowHeight;
943 void AddObject(ObjectInfo object, char * name, CodeObjectType type, bool select)
945 DataRow after = null;
947 CodeObject codeObject;
948 char * bitmap = null;
949 bool foundClass = false;
951 for(row = dropBox.firstRow; row; row = row.next)
953 CodeObject codeObject = row.GetData(null);
954 if(codeObject.object.oClass == object.oClass)
961 row = (DataRow)dropBox.AddRowAfter(after);
963 row.tag = (int)object;
970 indent = (type == typeClass) ? 0 : 1;
973 if(type != typeClass)
974 bitmap = (char *)eClass_GetProperty(object.instance._class, "icon");
977 codeObject.bitmap = { bitmap };
978 AddResource(codeObject.bitmap);
981 row.SetData(null, codeObject);
985 this.object = object ? object.instance : null;
986 propertyValue.userData = object ? (void *)object.instance : null;
987 dropBox.SelectRow(row);
991 void DeleteObject(ObjectInfo object)
993 DataRow row = dropBox.FindRow((int)object);
996 CodeObject codeObject = row.GetData(null);
998 if(codeObject.bitmap)
999 RemoveResource(codeObject.bitmap);
1000 dropBox.DeleteRow(row);
1004 void SelectObject(ObjectInfo object)
1008 DataRow row = dropBox.FindRow((int)object);
1009 this.object = object ? object.instance : null;
1010 propertyValue.userData = object ? (void *)object.instance : null;
1011 dropBox.SelectRow(row);
1015 void RenameObject(ObjectInfo object, char * name)
1017 DataRow row = dropBox.FindRow((int)object);
1018 CodeObject codeObject = row.GetData(null);
1019 // Isn't this useless? Shouldn't it be after?
1020 codeObject.name = name;
1021 // row.SetData(null, codeObject); // Is this necessary?
1024 void DataBox::EditSetData(void * setValue, bool closingDropDown)
1026 ((Sheet)master.master).SetData(setValue, this);
1029 void SetData(void * setValue, DataBox dataBox)
1031 //PropertyInfo propertyPtr = row.GetData(null);
1032 PropertyInfo propertyPtr = properties.GetData(null);
1033 Property prop = propertyPtr ? propertyPtr.prop : null;
1034 Instance object = this.object;
1037 Class dataType = prop.dataTypeClass;
1039 dataType = prop.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, prop.dataTypeString);
1040 if(propertyPtr.subMember)
1042 DataMember member = propertyPtr.subMember;
1043 Class subDataType = member.dataTypeClass;
1044 if(!member.dataTypeClass)
1045 subDataType = member.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, member.dataTypeString);
1049 if(!subDataType.dataType)
1050 subDataType.dataType = ProcessTypeString(subDataType.dataTypeString, false);
1052 if(dataType.type == structClass)
1054 data = new0 byte[dataType.structSize];
1055 prop.Get(object, data);
1056 // CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, &setValue, subDataType.size);
1057 CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, &setValue, subDataType.dataType.size);
1058 prop.Set(object, data);
1060 else if(dataType.type == normalClass || dataType.type == noHeadClass)
1065 if(dataType.type == bitClass)
1067 BitMember bitMember = (BitMember) member;
1070 DataValue value = { 0 };
1071 value.ui = prop.Get(object);
1072 value.ui &= ~ (uint)bitMember.mask;
1073 value.ui |= (uint)setValue << bitMember.pos;
1074 prop.Set(object, value.ui);
1079 data = dataType.typeSize ? new0 byte[dataType.typeSize] : null;
1080 prop.Get(object, data);
1081 // CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, &setValue, subDataType.typeSize);
1082 CopyBytes((byte *)data + member.offset + propertyPtr.extraOffset, &setValue, subDataType.dataType.size);
1083 // TODO: Support non 32 bit datatypes here
1084 prop.Set(object, data);
1088 if(data) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType,&data);
1092 else if(propertyPtr.subProperty)
1094 Property subProperty = propertyPtr.subProperty;
1095 Class subDataType = subProperty.dataTypeClass;
1097 subDataType = subProperty.dataTypeClass = eSystem_FindClass(codeEditor.privateModule, subProperty.dataTypeString);
1102 if(dataType.type == structClass)
1104 data = new0 byte[dataType.structSize];
1105 prop.Get(object, data);
1106 subProperty.Set(data, (uint)setValue);
1107 prop.Set(object, data);
1109 else if(dataType.type == normalClass || dataType.type == noHeadClass)
1111 Instance current = (Instance)prop.Get(object);
1112 Instance propObject = eInstance_New(dataType);
1113 CopyInstanceData(dataType, propObject, current);
1114 subProperty.Set(propObject, setValue);
1115 prop.Set(object, propObject);
1119 data = dataType.typeSize ? new0 byte[dataType.typeSize] : null;
1120 prop.Get(object, data);
1121 subProperty.Set(data, setValue);
1122 // TODO: Support not 32 bit data types here
1123 prop.Set(object, data);
1126 if(data) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType,&data);
1132 SetPropValue(prop, object, setValue);
1134 Code_FixProperty(propertyPtr.prop, object);
1136 properties.Update(null);
1137 dropBox.Update(null);
1138 codeEditor.designer.Update(null);
1139 codeEditor.Update(null); // patch for redraw bug if on top
1141 ListProperties(false);
1143 dataBox.editor.font = { propertyPtr.font.faceName, propertyPtr.font.size, propertyPtr.font.bold };
1144 codeEditor.ModifyCode();
1148 bool SaveEdit(PropertyInfo propertyPtr, Instance object)
1150 codeEditor.designer.Update(null);
1151 codeEditor.Update(null); // patch for redraw bug if on top
1152 properties.Update(null);
1153 dropBox.Update(null);
1155 Code_FixProperty(propertyPtr.prop, object);
1156 ListProperties(false);
1158 codeEditor.ModifyCode();
1164 if(!propBtn.checked)
1166 propBtn.checked = true;
1167 propBtn.NotifyClicked(this, propBtn, 0,0,0);
1171 methBtn.checked = true;
1172 methBtn.NotifyClicked(this, methBtn, 0,0,0);
1176 bool AttachMethodSelected(MenuItem selection, Modifiers mods)
1178 ClassFunction function = (ClassFunction)selection.id;
1179 codeEditor.AttachMethod(attachMethod, function);
1183 bool ReattachMethodSelected(MenuItem selection, Modifiers mods)
1185 ClassFunction function = (ClassFunction)selection.id;
1186 CodeObject object = methods.GetData(methodName);
1187 codeEditor.ReAttachMethod(attachMethod, function);
1191 bool OverrideMethodSelected(MenuItem selection, Modifiers mods)
1193 ClassFunction function = (ClassFunction)selection.id;
1194 CodeObject object = methods.GetData(methodName);
1196 codeEditor.AddMethod(object.method);
1200 bool GotoMethodSelected(MenuItem selection, Modifiers mods)
1202 ClassFunction function = (ClassFunction)selection.id;
1203 CodeObject object = methods.GetData(methodName);
1205 codeEditor.GoToMethod(object.method.name);
1209 bool DetachMethodSelected(MenuItem selection, Modifiers mods)
1211 ClassFunction function = (ClassFunction)selection.id;
1212 CodeObject object = methods.GetData(methodName);
1214 codeEditor.DetachMethod(object.method, object.function, object.overriden);
1218 bool DeleteMethodSelected(MenuItem selection, Modifiers mods)
1220 ClassFunction function = (ClassFunction)selection.id;
1221 CodeObject object = methods.GetData(methodName);
1223 object.deleteBtn.NotifyClicked(this, object.deleteBtn, 0,0,0);
1227 bool AddMethodClicked(Button button, int x, int y, Modifiers mods)
1229 DataRow row = (DataRow)button.id;
1230 CodeObject object = row.GetData(methodName);
1231 codeEditor.AddMethod(object.method);
1235 void CreateButtons(CodeObject codeObject, int y, int h, DataRow row)
1237 BitmapResource bitmap;
1239 if(codeObject.overriden)
1241 if(codeObject.overriden == 1)
1243 codeObject.deleteBtn = Button
1245 methods, master = this,
1247 bitmap = { ":actions/delete.png", alphaBlend = true },
1248 anchor = { right = 16, top = y },
1252 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
1254 CodeObject codeObject = ((DataRow)button.id).GetData(null);
1255 bool confirmation = !Code_IsFunctionEmpty(codeObject.function, codeObject.method, codeEditor.selected);
1260 sprintf(title, "Delete %s", codeObject.name);
1263 master = parent, type = okCancel, text = title,
1264 contents = "Method still contains code. Are you sure you want to delete it?"
1266 confirmation = false;
1269 if(!confirmation && codeObject.function.attached.count)
1272 sprintf(title, "Delete %s", codeObject.name);
1273 confirmation = true;
1276 master = parent, type = okCancel, text = title,
1277 contents = "Other methods are still attached to this method. Are you sure you want to delete it?"
1279 confirmation = false;
1284 codeEditor.DeleteMethod(codeObject.function);
1289 incref codeObject.deleteBtn;
1290 codeObject.deleteBtn.Create();
1293 if(codeObject.overriden == 2 || !codeObject.function.attached.count)
1295 codeObject.detachBtn = Button
1298 master = methods.master,
1300 bitmap = { ":actions/detach.png" },
1301 anchor = { right = 0, top = y },
1305 bool NotifyClicked(Button button, int x, int y, Modifiers mods)
1307 DataRow row = (DataRow)button.id;
1308 CodeObject object = row.GetData(methodName);
1310 codeEditor.DetachMethod(object.method, object.function, object.overriden);
1314 incref codeObject.detachBtn;
1315 codeObject.detachBtn.Create();
1320 if(codeObject.compatible.count)
1322 codeObject.attachBtn = Button
1324 parent = methods, master = methods.master,
1326 bitmap = { ":actions/attach.png" },
1327 anchor = { right = 0, top = y },
1331 bool NotifyPushed(Button button, int x, int y, Modifiers mods)
1334 DataRow row = (DataRow)button.id;
1335 CodeObject object = row.GetData(methodName);
1337 PopupMenu popupMenu;
1341 for(compatible = object.compatible.first; compatible; compatible = compatible.next)
1343 ClassFunction function = compatible.data;
1344 MenuItem { menu, function.declarator.symbol.string, id = (int)function, NotifySelect = AttachMethodSelected };
1346 attachMethod = object.method;
1348 popupMenu = PopupMenu
1350 master = this, menu = menu,
1353 button.absPosition.x - app.desktop.position.x,
1354 button.absPosition.y - app.desktop.position.y + button.size.h
1358 button.ReleaseCapture();
1359 popupMenu.Capture();
1363 incref codeObject.attachBtn;
1364 codeObject.attachBtn.Create();
1369 Method attachMethod;
1370 char selectedMethod[1024];
1371 CodeEditor codeEditor;
1373 char selectedProp[1024];
1377 static int String_OnCompare(char ** string1, char ** string2)
1380 if(*string1 && *string2)
1381 result = strcmpi(*string1, *string2);
1382 else if(!*string1 && *string2)
1384 else if(*string1 && !*string2)
1389 static void CopyInstanceData(Class dataType, Instance propObject, Instance current)
1392 for(_class = dataType; _class && _class.type != systemClass; _class = _class.base)
1395 for(member = _class.membersAndProperties.first; member; member = member.next)
1397 Class memberType = member.dataTypeClass;
1399 memberType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1400 if(member.isProperty)
1402 Property subProp = (Property) member;
1403 if(subProp.Get && subProp.Set)
1404 CopyProperty(subProp, propObject, current);
1409 // TOCHECK: I have serious doubts this works in many cases.
1410 memberType._vTbl[__ecereVMethodID_class_OnCopy](memberType, (byte *)propObject + member.offset, (byte *)current + member.offset);
1411 else if(member.memberOffset)
1412 memcpy((byte *)propObject + member.offset, (byte *)current + member.offset, member.memberOffset);
1418 class PropertyInfo : struct
1424 char * categoryName;
1425 DataMember subMember;
1426 Property subProperty;
1429 void OnDisplay(Surface surface, int x, int y, int width, Instance object, Alignment alignment, DataDisplayFlags displayFlags)
1431 Property prop = this.prop;
1433 surface.TextFont(font.font);
1436 surface.SetBackground(Color { 170, 170, 170 });
1437 surface.Area(0,0, x+width-1, y+100);
1439 else if(prop && prop.dataTypeString)
1441 Class dataType = prop.dataTypeClass;
1442 Module module = ((Designer)GetActiveDesigner()).codeEditor.privateModule;
1444 dataType = prop.dataTypeClass = eSystem_FindClass(module, prop.dataTypeString);
1446 if(dataType && prop.Get)
1448 void * dataPtr, * data = null, * subData = null;
1449 DataValue valueData, valueSubData;
1453 if(dataType.type == structClass)
1455 data = new0 byte[dataType.structSize];
1456 prop.Get(object, data);
1461 GetProperty(prop, object, &valueData);
1462 if(dataType.type == normalClass)
1463 dataPtr = valueData.p;
1465 dataPtr = &valueData;
1471 DataMember member = this.subMember;
1472 Class subDataType = member.dataTypeClass;
1474 subDataType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1477 if(dataType.type == bitClass)
1479 BitMember bitMember = (BitMember)member;
1480 bitValue = (valueData.i & bitMember.mask) >> bitMember.pos;
1481 dataPtr = &bitValue;
1484 dataPtr = (byte *)dataPtr + member.offset + this.extraOffset;
1486 dataType = subDataType;
1488 else if(this.subProperty)
1490 Property subProperty = this.subProperty;
1491 Class subDataType = subProperty.dataTypeClass;
1493 subDataType = subProperty.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, subProperty.dataTypeString);
1494 if(!subProperty.Get) subDataType = null;
1497 if(subDataType.type == structClass)
1499 subData = new0 byte[subDataType.structSize];
1500 subProperty.Get(dataPtr, subData);
1505 GetProperty(subProperty, dataPtr, &valueSubData);
1506 if(subDataType.type == normalClass)
1507 dataPtr = valueSubData.p;
1509 dataPtr = &valueSubData;
1512 dataType = subDataType;
1516 dataType._vTbl[__ecereVMethodID_class_OnDisplay](dataType, dataPtr, surface, x, y, width, null, alignment, displayFlags);
1524 Window OnEdit(DataBox dataBox, Window obsolete, int x, int y, int w, int h, void * unused)
1526 EditBox editData = null;
1527 Property prop = this.prop;
1529 dataBox.SetData = Sheet::EditSetData;
1530 if(prop && prop.dataTypeString && !this.disabled)
1532 Sheet propertyWindow = (Sheet)dataBox.master.master;
1533 Instance object = propertyWindow.object;
1534 Class dataType = prop.dataTypeClass;
1536 dataType = prop.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, prop.dataTypeString);
1538 if(dataType && prop.Get)
1540 void * dataPtr, * data = null, * subData = null;
1541 DataValue valueData, valueSubData;
1545 if(dataType.type == structClass)
1547 data = new0 byte[dataType.structSize];
1548 prop.Get(object, data);
1553 GetProperty(prop, object, &valueData);
1554 if(dataType.type == normalClass)
1555 dataPtr = valueData.p;
1557 dataPtr = &valueData;
1563 DataMember member = this.subMember;
1564 Class subDataType = member.dataTypeClass;
1566 subDataType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1569 if(dataType.type == bitClass)
1571 BitMember bitMember = (BitMember)member;
1572 bitValue = (valueData.i & bitMember.mask) >> bitMember.pos;
1573 dataPtr = &bitValue;
1576 dataPtr = (byte *)dataPtr + member.offset + this.extraOffset;
1578 dataType = subDataType;
1580 else if(this.subProperty)
1582 Property subProperty = this.subProperty;
1583 Class subDataType = subProperty.dataTypeClass;
1585 subDataType = subProperty.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, subProperty.dataTypeString);
1586 if(!subProperty.Get) subDataType = null;
1589 if(subDataType.type == structClass)
1591 subData = new0 byte[subDataType.structSize];
1592 subProperty.Get(dataPtr, subData);
1597 GetProperty(subProperty, dataPtr, &valueSubData);
1598 if(subDataType.type == normalClass)
1599 dataPtr = valueSubData.p;
1601 dataPtr = &valueSubData;
1604 dataType = subDataType;
1608 editData = (void *)dataType._vTbl[__ecereVMethodID_class_OnEdit](dataType, dataPtr, dataBox, obsolete, x, y, w, h, object /*unused*/);
1613 editData.font = { font.faceName, font.size, font.bold };
1619 int OnCompare(PropertyInfo data2)
1621 char * category1 = prop ? prop.category : categoryName;
1622 char * category2 = data2.prop ? data2.prop.category : data2.categoryName;
1625 if(!category1) category1 = "Misc";
1626 if(!category2) category2 = "Misc";
1630 // result = String::OnCompare((String)category1, (String)category2);
1631 result = String_OnCompare(&category1, &category2);
1636 if(subMember && !data2.subMember)
1640 else if(!subMember && data2.subMember)
1644 else if(subMember && data2.subMember)
1646 if(subMember.id < data2.subMember.id)
1648 else if(subMember.id > data2.subMember.id)
1653 else if(subProperty && !data2.subProperty)
1657 else if(!subProperty && data2.subProperty)
1661 else if(subProperty && data2.subProperty)
1663 if(subProperty.id < data2.subProperty.id)
1665 else if(subProperty.id > data2.subProperty.id)
1670 else if(prop && !data2.prop)
1672 else if(!prop && data2.prop)
1675 // result = ((String)prop.name).OnCompare(data2.prop.name);
1676 // result = String::OnCompare((String)prop.name, (String)data2.prop.name);
1677 result = String_OnCompare(&prop.name, &data2.prop.name);
1682 bool OnSaveEdit(Window editControl, void * unusedData)
1684 Property prop = this.prop;
1687 Sheet sheet = (Sheet)editControl.master.master.master;
1688 Instance object = sheet.object;
1689 Class mainDataType = prop.dataTypeClass;
1691 bool result = false;
1692 void * dataPtr, * data = null, * subData = null;
1693 void * propObject = null;
1694 DataValue valueData = { 0 }, valueSubData = { 0 };
1698 mainDataType = prop.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, prop.dataTypeString);
1699 dataType = mainDataType;
1701 // Prepare main prop
1702 if(dataType.type == structClass)
1704 data = new0 byte[dataType.structSize];
1705 if(this.subMember || this.subProperty)
1706 prop.Get(object, data);
1710 else if(dataType.type == normalClass || dataType.type == noHeadClass)
1712 dataPtr = &valueData;
1714 if(this.subMember || this.subProperty)
1717 Instance current = (Instance)prop.Get(object);
1718 propObject = valueData.p = eInstance_New(dataType);
1719 CopyInstanceData(dataType, propObject, current);
1725 if(this.subMember || this.subProperty)
1726 GetProperty(prop, object, &valueData);
1728 dataPtr = &valueData;
1729 propObject = &valueData;
1735 DataMember member = this.subMember;
1736 Class subDataType = member.dataTypeClass;
1738 subDataType = member.dataTypeClass = eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, member.dataTypeString);
1741 if(dataType.type == bitClass)
1742 dataPtr = &bitValue;
1744 dataPtr = (byte *)dataPtr + member.offset + this.extraOffset;
1746 dataType = subDataType;
1748 else if(this.subProperty)
1750 Property subProperty = this.subProperty;
1751 Class subDataType = subProperty.dataTypeClass;
1754 subDataType = subProperty.dataTypeClass =
1755 eSystem_FindClass(((Designer)GetActiveDesigner()).codeEditor.privateModule, subProperty.dataTypeString);
1756 if(!subProperty.Get) subDataType = null;
1759 if(subDataType.type == structClass)
1761 subData = new0 byte[subDataType.structSize];
1765 dataPtr = &valueSubData;
1767 dataType = subDataType;
1772 if(dataType._vTbl[__ecereVMethodID_class_OnSaveEdit](dataType, dataPtr, editControl, null))
1774 if(mainDataType.type == bitClass && this.subMember)
1776 BitMember bitMember = (BitMember)this.subMember;
1777 valueData.ui &= ~ (uint)bitMember.mask;
1778 valueData.ui |= bitValue << bitMember.pos;
1780 if(this.subProperty)
1782 if(dataType.type == structClass)
1783 this.subProperty.Set(propObject, subData);
1784 else if(dataType.type == unitClass || dataType.type == enumClass || dataType.type == bitClass)
1786 if(!strcmp(dataType.dataTypeString, "float"))
1787 ((void(*)(void *,float))(void *)this.subProperty.Set)(propObject, valueSubData.f);
1788 else if(!strcmp(dataType.dataTypeString, "double"))
1789 ((void(*)(void *,double))(void *)this.subProperty.Set)(propObject, valueSubData.d);
1790 else if(!strcmp(dataType.dataTypeString, "byte"))
1791 ((void(*)(void *,byte))(void *)this.subProperty.Set)(propObject, valueSubData.uc);
1792 else if(!strcmp(dataType.dataTypeString, "uint16"))
1793 ((void(*)(void *,uint16))(void *)this.subProperty.Set)(propObject, valueSubData.us);
1795 this.subProperty.Set(propObject, valueSubData.ui);
1798 this.subProperty.Set(propObject, valueSubData.ui);
1800 if(mainDataType.type == structClass)
1801 prop.Set(object, data);
1802 else if(mainDataType.type == unitClass || mainDataType.type == enumClass || mainDataType.type == bitClass)
1804 if(!strcmp(mainDataType.dataTypeString, "float"))
1805 ((void(*)(void *,float))(void *)prop.Set)(object, valueData.f);
1806 else if(!strcmp(mainDataType.dataTypeString, "double"))
1807 ((void(*)(void *,double))(void *)prop.Set)(object, valueData.d);
1808 else if(!strcmp(mainDataType.dataTypeString, "byte"))
1809 ((void(*)(void *,byte))(void *)prop.Set)(object, valueData.uc);
1810 else if(!strcmp(mainDataType.dataTypeString, "uint16"))
1811 ((void(*)(void *,uint16))(void *)prop.Set)(object, valueData.us);
1813 prop.Set(object, valueData.ui);
1816 prop.Set(object, valueData.ui);
1820 if(data == dataPtr) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType, &data);
1821 if(subData == dataPtr) dataType._vTbl[__ecereVMethodID_class_OnFree](dataType, &subData);
1827 return sheet.SaveEdit(this, object);
1833 class Category : struct
1835 Category prev, next;