-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathCDMLib.py
3232 lines (2450 loc) · 121 KB
/
CDMLib.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
###############################################################################
# Copyright (C) 2013-2018 Jacob Barhak
# Copyright (C) 2009-2012 The Regents of the University of Michigan
#
# This file is part of the MIcroSimulation Tool (MIST).
# The MIcroSimulation Tool (MIST) is free software: you
# can redistribute it and/or modify it under the terms of the GNU General
# Public License as published by the Free Software Foundation, either
# version 3 of the License, or (at your option) any later version.
#
# The MIcroSimulation Tool (MIST) is distributed in the
# hope that it will be useful, but WITHOUT ANY WARRANTY; without even the
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
# See the GNU General Public License for more details.
###############################################################################
#
# ADDITIONAL CLARIFICATION
#
# The MIcroSimulation Tool (MIST) is distributed in the
# hope that it will be useful, but "as is" and WITHOUT ANY WARRANTY of any
# kind, including any warranty that it will not infringe on any property
# rights of another party or the IMPLIED WARRANTIES OF MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. THE AUTHORS assume no responsibilities
# with respect to the use of the MIcroSimulation Tool (MIST).
#
# The MIcroSimulation Tool (MIST) was derived from the Indirect Estimation
# and Simulation Tool (IEST) and uses code distributed under the IEST name.
# The change of the name signifies a split from the original design that
# focuses on microsimulation. For the sake of completeness, the copyright
# statement from the original tool developed by the University of Michigan
# is provided below and is also mentioned above.
#
###############################################################################
############################ Original Copyright ###############################
###############################################################################
# Copyright (C) 2009-2012 The Regents of the University of Michigan
# Initially developed by Deanna Isaman, Jacob Barhak, Donghee Lee
#
# This file is part of the Indirect Estimation and Simulation Tool (IEST).
# The Indirect Estimation and Simulation Tool (IEST) is free software: you
# can redistribute it and/or modify it under the terms of the GNU General
# Public License as published by the Free Software Foundation, either
# version 3 of the License, or (at your option) any later version.
#
# The Indirect Estimation and Simulation Tool (IEST) is distributed in the
# hope that it will be useful, but WITHOUT ANY WARRANTY; without even the
# implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
# See the GNU General Public License for more details.
################################################################################
# #
# This file contains all common constants, classes and functions for GUI #
################################################################################
import os
# There is an issue with overlay scroll bars on Ubuntu Linux that breaks
# combo controls. This line disables these scroll bars in favor of old style
# scroll bars that are more comppatible.
os.environ['LIBOVERLAY_SCROLLBAR']='0'
import wxversion
# This line ensures that all wx imports will have a proper version
# therefore it is important that this file is always imported before wx
wxversion.ensureMinimal("3.0")
import wx
import wx.combo
import wx.lib.buttons
from wx.lib.wordwrap import wordwrap
import sys
import string
import types
import cStringIO
import thread
import DataDef as DB
import HelpInterface
import copy
# Define the GUI version
Version = (0,92,5,0,'MIST')
CompatibleWithDataDefVersion = (0,92,5,0,'MIST')
# Check that the Data Version and the GUI version are compatible
# Currently the Last text identifier in the version is ignored in
# comparison - this may change in the future.
if not hasattr(DB, 'Version'):
VersionOfDB = None
else:
VersionOfDB = DB.Version
if VersionOfDB == None or VersionOfDB[0:4] != CompatibleWithDataDefVersion[0:4]:
raise ValueError, ' GUI version ' + str(Version) + ' is compatible only with DataDef version: ' + str (CompatibleWithDataDefVersion) + ' or higher, while attempting to load DataDef version: ' + str(VersionOfDB)
# CONSTANTS used in Project
# Ids used in menus
ID_MENU_ABOUT = wx.NewId()
ID_MENU_HELP = wx.NewId()
ID_MENU_HELP_GENERAL = wx.NewId()
ID_MENU_REPORT_THIS = wx.NewId()
ID_MENU_REPORT_ALL = wx.NewId()
# This is used by a popup menu
ID_MENU_COPY_RECORD = wx.NewId()
ID_SPECIAL_RECORD_ADD = wx.NewId()
ID_MENU_DELETE_RECORD = wx.NewId()
# Button ids used in frame
IDF_BUTTON1 = wx.NewId()
IDF_BUTTON2 = wx.NewId()
IDF_BUTTON3 = wx.NewId()
IDF_BUTTON4 = wx.NewId()
IDF_BUTTON5 = wx.NewId()
IDF_BUTTON6 = wx.NewId()
IDF_BUTTON7 = wx.NewId()
IDF_BUTTON8 = wx.NewId()
IDF_BUTTON9 = wx.NewId()
IDF_BUTTON10 = wx.NewId()
IDF_BUTTON11 = wx.NewId()
IDF_BUTTON12 = wx.NewId()
IDF_BUTTON13 = wx.NewId()
IDF_BUTTON14 = wx.NewId()
IDF_BUTTON15 = wx.NewId()
IDF_BUTTON16 = wx.NewId()
IDF_BUTTON17 = wx.NewId()
IDF_BUTTON18 = wx.NewId()
IDF_BUTTON19 = wx.NewId()
IDF_BUTTON20 = wx.NewId()
# Button ids used in panel
# Actually following constants are defined for convenience.
IDP_BUTTON1 = wx.NewId()
IDP_BUTTON2 = wx.NewId()
IDP_BUTTON3 = wx.NewId()
IDP_BUTTON4 = wx.NewId()
IDP_BUTTON5 = wx.NewId()
IDP_BUTTON6 = wx.NewId()
IDP_BUTTON7 = wx.NewId()
IDP_BUTTON8 = wx.NewId()
IDP_BUTTON9 = wx.NewId()
IDP_BUTTON10 = wx.NewId()
# Data type definition for Control class
ID_TYPE_NONE = -1
ID_TYPE_ALPHA = 0
ID_TYPE_INTEG = 1
ID_TYPE_FLOAT = 2
ID_TYPE_COMBO = 3
# Constants for event
ID_EVT_SORT = wx.NewId() # dedicated id for sort.
# If this flag is set, default event handler call SortPanel function
ID_EVT_OWN = wx.NewId() # indicat this control has own event handler.
# it is called by FrameEventHandler in CDMFrame class
ID_EVT_OPEN_WINDOW = wx.NewId() # Not used currently
# Ids for KeyValidator class
ALPHA_ONLY = 1 # only allow characters
DIGIT_ONLY = 2 # only allow numbers, including +/- sign
NO_INPUT = 0 # not allow any keyboard input
NO_EDIT = 3 # Not allow del or backspace key
## Id for Project Type
#ID_PRJ_SIMULATION = 'Simulation'
#ID_PRJ_ESTIMATION = 'Estimation'
# Id for open mode of a form
# NONE : no specific mode. Thus, all objects will be displaced in the scrolled window
# multi : in other word, 'filtered' mode. need only for 'Transition' form in current version
# single : Display single object to edit/create new object in any collection except Transition
ID_MODE_NONE = None
ID_MODE_MULTI = 'multi'
ID_MODE_SINGL = 'single'
# define supported commands for focus switch
SUPPORTED_COMMANDS = [ wx.wxEVT_COMMAND_BUTTON_CLICKED,
wx.wxEVT_COMMAND_LEFT_CLICK,
wx.wxEVT_COMMAND_MENU_SELECTED,
wx.wxEVT_COMMAND_LIST_ITEM_SELECTED ]
# common functions used in the CDM library and forms
def iif(condition, true_value, false_value):
"""Immediate if: if statement is true return TruePart otherwise FalsePart
This function is copied from DataDef.py
"""
if condition: return true_value
else: return false_value
def xproperty(fget, fset, fdel=None, doc=None):
""" Map set/get method to a property.
Thus, 'form.SetProperty(PropertyA = value)' is same as 'form.PropertyA = value'
"""
if isinstance(fget, str):
attr_name = fget
def fget(obj): return getattr(obj, attr_name)
if isinstance(fset, str):
attr_name = fset
def fset(obj, val): setattr(obj, attr_name, val)
return property(fget, fset, fdel, doc)
def Exist(obj):
""" Simple function to check if an object is None or not """
return obj != None
def OpenPopupMenu(self, event=None):
""" Open Popup menu. To use this function, a frame should have a menu list named 'pup_menus' """
if not hasattr(self, 'pup_menus'): return
menu = setupMenu(self, self.pup_menus, False) # crate popup menu and assign event handler
self.PopupMenu(menu) # open popup menu
menu.Destroy() # remove from memory to show just once when right button is clicked
def CloseForm(self, refresh=True, collection=None, key=None):
""" Common function to close new form """
# check if current form has parent form
parent = self.GetParent()
self.MyMakeModal(False)
self.Destroy()
# If entered data is ok, throw event to refresh to parent row panel
if refresh :
SetRefreshInfo(self, collection, key) # save data(i.e. collection, key) to refresh field
parent = self.GetParent()
if parent!=None:
event = wx.PyEvent() # create new event
event.SetEventType(wx.wxEVT_END_PROCESS)
wx.PostEvent(parent, event) # throw the event to the parent of current frame
if parent is None : return
if not parent.IsTopLevel(): parent = parent.GetTopLevelParent()
# if current form has parent form, raise it to the top of the screen
parent.MyMakeModal() # MakeModal(False) command makes all forms as Non-modal forms
# Thus, before raising the parent, MakeModal method should be called once more
parent.Raise()
def OpenForm(name_module, parent, mode=None, key=None, type=None, id_prj=0):
""" Common function to open a form
arguments
name_module : module name that includes the form
parent : parent of new form, should be an instance of wx.Window class
mode : Set the initial mode of new form.Default mode is None that displays all the data in a collection
key : key(usually ID) of the record that is used in new form
type : Specific type of record if needed. Additional data for initialization. Type of the target is changed according to mode
See Initialization method in CDMFrame class
id_prj : ID of a project. It is used to prevent data change.
"""
module = __import__(name_module) # import form module
form = module.MainFrame(mode, key, type, id_prj, parent)# create an instance of the form
form.MyMakeModal() # Make this frame as modal window
# Adjust the position of new form according to the screen size
# If the size of form is smaller than screen size, open new form on the center of screen
# Otherwise, set top left corner of the form at the corner of the screen
sw, sh = wx.GetDisplaySize()
fx, fy, fw, fh = form.GetRect()
ox = iif( sw < fw, 0, 0.5 * (sw-fw) )
oy = iif( sh < fh, 0, 0.5 * (sh-fh) )
wid = iif( sw < fw, sw, fw )
hgt = iif( sh < fh, sh, fh )
form.SetRect((ox,oy, wid, hgt))
form.Show()
return form
# Define Administrator mode by using a global list
AdminMode = [False]
def GetAdminMode():
""" Returns True if AdminMode set set """
return AdminMode[0]
def SetAdminMode(NewValue = False):
""" Set AdminMode to specified value """
AdminMode[0] = NewValue
# global variable to save/retrieve refresh data
_com_stack = []
def GetRefreshInfo():
""" Retrieve a database object using data in the clipboard
This function is used to refresh information after closing child form
"""
global _com_stack
object = None
if len(_com_stack) == 0 :
return None
info = _com_stack.pop()
if info[0] == '':
object = info[1] # In current version, this line is used only for Cost/QoL Wizard
elif info[1] is None:
object = info[1]
else:
object = GetRecordByKey(getattr(DB, info[0]), info[1])
return object
def SetRefreshInfo(self, Collection=None, Key=None):
""" Set target information for refreshing in the clipboard"""
global _com_stack
success = not (Collection == None and Key == None)
if success:
_com_stack.append((Collection, Key))
return success
class Struct:
""" Dummy class to create structure-like data
It is used to replicate the variables in each database object
See 'GetInstanceAttr' function
"""
pass
def OpenAbout(self):
""" Display information about Chronic Disease Model"""
info = wx.AboutDialogInfo()
info.Name = "MIcro Simulation Tool (MIST)\n"
info.Version = 'Data Definitions Version: ' + str(DB.Version) + '\n GUI Version: '+ str(Version)
info.Copyright = " Copyright (C) 2013-2014 Jacob Barhak\n Copyright (C) 2009-2012 The Regents of the University of Michigan (IEST)"
info.Description = wordwrap(
"The Micro-simulation tools is a Monte-Carlo simulation compiler"
"It has been initially designed to help model Chronic Diseases."
"MIST is a split branch from the GPL code of the "
"Indirect Estimation and Simulation Tool (IEST). "
"More information is available on the web site of the developer. ",
350, wx.ClientDC(self))
info.WebSite = ("http://sites.google.com/site/jacobbarhak/",
"")
info.Developers = [ "\n Jacob Barhak - MIST system design and implementation"
"\n see IEST documentation for additional developers"]
info.License = wordwrap(" The MIcroSimulation Tool (MIST) is free software: you"
" can redistribute it and/or modify it under the terms of the GNU General"
" Public License as published by the Free Software Foundation, either"
" version 3 of the License, or (at your option) any later version."
" \n"
" The MIcroSimulation Tool (MIST) is distributed in the"
" hope that it will be useful, but WITHOUT ANY WARRANTY; without even the"
" implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE."
" See the GNU General Public License for more details."
,350, wx.ClientDC(self) )
wx.AboutBox(info)
#-----------------------------------------------------------------------#
# Base class for all window class used in GUI #
# Event and blank properties are defined #
#-----------------------------------------------------------------------#
class CDMWindow(wx.Window):
""" Base Window class for CDM project. Derived from wx.Window class
Base properties of a CDMWindow class are defined here
See FrameEventHandler method in CDMFrame class for the usage of the event properties"""
def __init__(self):
self._isEmpty = False # flag to check empty panel
self._evtType = None # event type : reserved for future extension
self._evtID = None # event ID : define specific action
self._evtData = None # data for event : depends on the event ID
self._user_data = None # user(programmer) definable data,
# setters for above variables
def SetEvtType(self, evt_type):
""" Setter Method for Event Type of a CDMWindow Instance"""
self._evtType = evt_type
def SetEvtData(self, evt_data):
""" Setter Method for Event Data of a CDMWindow Instance"""
self._evtData = evt_data
def SetEvtID(self, evt_id):
""" Setter Method for Event Id of a CDMWindow Instance"""
self._evtID = evt_id
def SetEvent(self, evt):
""" Setter Method for Event Related Properties (Type, Id, Data) of a CDMWindow Instance"""
self._evtType = evt[0]
self._evtID = evt[1]
self._evtData = evt[2]
def SetUserData(self, user_data):
""" Setter Method for User Data of a CDMWindow Instance"""
if type(self._user_data) == list: # if type of user data is list
self._user_data.append(user_data)
else: # otherwise
self._user_data = user_data
def SetBackgroundColor(self, bgcolor=None):
""" Setter Method for Background Color of a CDMWindow Instance"""
if bgcolor == None:
bgcolor = wx.SystemSettings_GetColour(wx.SYS_COLOUR_3DFACE)
self.SetBackgroundColour(bgcolor)
# Map the setter methods to a property
# Using 'xproperty' function,
# Getter/Setter methods can be used as same form
# Ex : win.evtData = data is same as win.SetEvtData(data)
# evtType is not used in the version. It is reserved for the future extension.
evtType = xproperty( '_evtType', SetEvtType )
evtData = xproperty( '_evtData', SetEvtData )
evtID = xproperty( '_evtID', SetEvtID )
userData = xproperty( '_user_data', SetUserData )
def MyMakeModal(self, modal=True):
if self.IsTopLevel():
for w in wx.GetTopLevelWindows():
if w is not self and 'Inspection' not in str(w):
w.Enable(not modal)
#-----------------------------------------------------------------------#
# Base class for a frame. #
# Common properties and methods are defined #
#-----------------------------------------------------------------------#
class CDMFrame(CDMWindow, wx.Dialog, wx.Frame):
#class CDMFrame(CDMWindow, wx.Frame):
"""
Base Frame class for CDM project. It is derived from CDMWindow and wx.Frame class
Common properties, methods and event handler are defined in this class
"""
def __init__(self, mode=None, data=None, type=None, *args, **kwds):
""" Initialization method for CDMFrame class
Arguments
- mode : open mode (single, multi, None)
- data : used if mode is ID_MODE_SINGL. Usually, ID of a record
- type : used for data checking.
"""
CDMWindow.__init__(self) # Initialize super class
kwdsnew = copy.copy(kwds)
kwdsnew["style"] = wx.DEFAULT_FRAME_STYLE|wx.TAB_TRAVERSAL|wx.CLIP_CHILDREN # set window style
wx.Frame.__init__(self, *args, **kwdsnew)
self._id_prev_panel = -1 # define ids to target panels and controls
self._id_prev_ctrl = -1 # used to check the movement of focus
self._id_this_panel = -1
self._id_this_ctrl = -1
self._no_row = 1 # initialize no. of RowPanel instances in the scrolled window
self._collection = None # target collection(==name of global variables) in database related to this frame(or form)
self._open_mode = mode # Open mode of a form. It decides the number of RowPanel objects in ScrolledWindow
self._open_data = data # Data for opening mode(i.e. ID of a record)
self._open_type = type #
self.SetBackgroundColour(wx.SystemSettings_GetColour(wx.SYS_COLOUR_3DFACE))
# Bind common event handlers for the instances of CDMFrame class
self.Bind(wx.EVT_MOUSEWHEEL, self.OnMouseWheel) # Bind event handler for mouse wheel -> need to fix
self.Bind(wx.EVT_CONTEXT_MENU, self.OnContextMenu) # Bind event handler for right mouse click. Open Popup menu
self.Bind(wx.EVT_CLOSE, self.FrameEventHandler) # Bind event handler for close window event
# create new panel and assign event hander
def AddPanel(self , data = None):
""" Method to add new RowPanel instance.
Create and Set the position of new RowPanel instance"""
# First, check if there is a RowPanel which is being focused( open or new )
panel = self.FindRowPanelByStatus([wx.ID_OPEN, wx.ID_NEW])
if panel :
panel.SetStatus(wx.ID_NONE)
new_panel = self.GetBlankPanel()
if new_panel==None:
# get position and size of last RowPanel
# to calculate the position of new row panel
x, y, w, h = 0, 0, 0, 0
panels = list(self.pn_view.GetChildren())
if panels:
x, y, w, h = panels[-1].GetRect()
# add new RowPanel defined in module
if hasattr(self, 'SetupPanel'):
new_panel = self.SetupPanel(py=y+h) # Dedicated Setup method for each form
# Need to be implemented in each module
else:
module = __import__(self.__module__)
new_panel = module.RowPanel(parent=self.pn_view, id=0, pos=(0,y+h))
self.pn_view.GetSizer().Add(new_panel, 0, wx.ALL, 1)
# Assign default event handler to controls in new RowPanel
self.BindDefaultEvent(new_panel)
# Find first editable control - Textbox, combobox or checkbox
# Then set focus in that control
new_ctrl = new_panel.GetDefaultFocus()
new_ctrl.SetFocus()
wx.CallAfter(adjustScrollBar, self, self.pn_view) # Resize the length of scroll bar according to the number of RowPanel instances
wx.CallAfter(scrollPanel, self.pn_view, new_panel) # Display new RowPanel instance
else:
# Find first editable control - Textbox, combobox or checkbox
# Then set focus in that control
new_ctrl = new_panel.GetDefaultFocus()
new_ctrl.SetFocus()
# If current RowPanel intance include Combo Control(s), it should have 'SetComboItem' method
if hasattr(new_panel, 'SetComboItem'):
new_panel.SetComboItem() # check items in a combobox and set items if necessary
if data == None:
# Set status of new RowPanel. Display asterisk(*) at the left side of the panel
new_panel.SetStatus(wx.ID_NEW)
else:
# In case data exists for the record, initialize the fields
new_panel.SetValues(data) # display current record data
new_panel.SetId(self.NextRowID)
self.IncrNextRowID() # increase row no for indexing
new_panel.SetStatus(wx.ID_OPEN)
new_panel.SaveValues() # copy panel data to record
# Reset target information again to reflect changes
# Reset target information
self.SetPrevTarget( new_panel, new_ctrl )
self.SetCurrTarget( new_panel, new_ctrl )
# bind event for all controls in the new panel
# ALL EVT_LEFT_UP evnts are for focus check
def BindDefaultEvent(self, new_panel):
""" Assign Event Handler to controls in a row panel """
for ctrl in new_panel.GetChildren(): # for all controls in new panel
type_ctrl = type(ctrl)
# Static text doesn't need event handler
if type_ctrl == wx._controls.StaticText or not ctrl.IsShown() :
continue
if type_ctrl in [ Button, BitmapButton, wx.Button ]:
ctrl.Bind(wx.EVT_BUTTON, self.FrameEventHandler)
elif type_ctrl == List:
ctrl.Bind(wx.EVT_LIST_ITEM_SELECTED, self.FrameEventHandler)
ctrl.Bind(wx.EVT_LIST_COL_CLICK, self.FrameEventHandler)
else:
ctrl.Bind(wx.EVT_LEFT_UP, self.FrameEventHandler) # for focus check
# if curren control is ComboBox, Bind the Text Control to check the focus change
if type_ctrl == Combo:
ctrl.GetTextCtrl().Bind(wx.EVT_LEFT_UP, self.FrameEventHandler)
# If a control has specific event type, bind this control to event handler for the event
# This code reserved for evtType property in CDMWindow. Not used currently
if Exist(ctrl.evtType):
ctrl.Bind(ctrl.evtType, self.FrameEventHandler)
# Bind this panel to Default Event Hanlder to check the focus change
new_panel.Bind(wx.EVT_LEFT_DOWN, self.FrameEventHandler)
def ForceRecordSaveAttempt(self):
""" Tries to force a record save to be performed if needed """
# Find current/previous panel and control
# The function returns False if an Error occurred and True otherwise
this_panel, this_ctrl = self.GetCurrTarget('obj')
try:
self.CheckFocus(this_panel, this_ctrl, this_panel, this_ctrl, None, True)
return True
except:
dlgErrorMsg(Parent = self)
return False
# check the focus change
def CheckFocus(self, prev_panel, prev_ctrl, this_panel, this_ctrl, typeEvt, ForceSaveAttempt = False):
""" Method to process focus change in an Instance of CDMFrame class"""
type_this_panel = iif( this_panel == None, this_panel, type(this_panel))
type_prev_panel = iif( prev_panel == None, prev_panel, type(prev_panel))
# check the focus moves from the title section to frame
if type_this_panel == type_prev_panel and type_this_panel == None : return
save_data = ForceSaveAttempt
if type_this_panel == type_prev_panel: # row panel <-> row panel or row panel <-> title
if this_panel.Id == prev_panel.Id :# if focus was moved in same panel
if hasattr(this_ctrl, 'chkFocus') and this_ctrl.chkFocus : # and current control needs to be checked
save_data = True
elif prev_panel.isRow :
save_data = True
else: # row panel <-> frame
if hasattr(prev_panel, 'isRow') and prev_panel.isRow:
save_data = True
if save_data :
record = prev_panel.GetValues() # read current values in previous panel
if not DB.IsEqualDetailed(record, prev_panel.record):
# Create new instance and save it to a collection
# *** Need to be implemented in each row panel ***
entry = prev_panel.SaveRecord(record)
# display relevant message according to the entry.
# If there is an error, current field won't be refreshed
if entry is None:
if type(self.openType) in [ tuple, list ]:
str_type = str(self.openType)
else:
str_type = self.openType
raise ValueError, 'INVALID type. Type of current record should be (or should be one of): "' + string.upper(str_type) + '"'
# if saved successfully and prev_panel is new, assign new Id
if prev_panel.Id == 0:
prev_panel.SetId(self.NextRowID)
self.IncrNextRowID() # increase row no for indexing
# Refresh screen using new entry
prev_panel.SetValues(entry) # *** Need to be implemented in each row panel ***
prev_panel.SaveValues() # saves the new data in the record info - this is used if the panel is not exited since a button is pressed
prev_panel.ClearComboItem() # release memory for ComboCtrls in prev_panel
if this_panel != None:
if this_panel.isRow:
# check items in a combobox and set items if necessary
if hasattr(this_panel, 'SetComboItem'):
this_panel.SetComboItem() # *** Need to be implemented in each row panel if necessary ***
# If focus is moved from title section or frame or dialog,
# save current values
if prev_panel is None or \
( prev_panel and prev_panel.Id != this_panel.Id ):
this_panel.SaveValues() # Copy current value to record property.
# *** Need to be implemented in each row panel if necessary ***
if type_prev_panel: # If previous panel is row panel
prev_panel.SetStatus( wx.ID_NONE )
else: # or check the panels in the scrolled window
panels = self.pn_view.GetChildren()
for panel in panels:
if panel.Id == this_panel.Id or panel.Status != wx.ID_OPEN: continue
panel.SetStatus( wx.ID_NONE )
break
if this_panel.isRow :
this_panel.SetStatus(this_panel.Id, False) # set status of current RowPanel
self.SetCurrTarget(this_panel, this_ctrl) # set current panel/control as previous objects
self.SetPrevTarget(this_panel, this_ctrl)
# Default event handler for a frame
# Basic functions - add, delete, focus change, undo, find - are done here
# Specific event defined for each control also can be done here
# ALL METHOD MARKED WITH STARS(***) SHOULD BE IMPLEMENTED IN EACH RowPanel Class
def FrameEventHandler(self, evt):
""" Default Event Handler of CDMFrame class """
evtType = evt.GetEventType()
this_ctrl = evt.GetEventObject()
menuId = evt.GetId()
try:
panels = list(self.pn_view.GetChildren())
# Find current/previous panel and control
this_panel, this_ctrl = self.InitCurrTarget(this_ctrl)
prev_panel, prev_ctrl = self.GetPrevTarget('obj')
# First check event(s) which doesn't require focus check such as Undo
# Undo function always use previous panel as target panel
# because if there was no error, prev_panel is same as this_panel
if menuId == wx.ID_UNDO:
prev_panel.Undo(getattr(DB, self.Collection))
return
# If row panels exist in the scrolled window, check focus
if panels != []:
self.CheckFocus(prev_panel, prev_ctrl, this_panel, this_ctrl, evtType)
# if current event is close window, call CloseForm function
if evtType == wx.wxEVT_CLOSE_WINDOW:
if not this_panel:
panel = prev_panel
else:
panel = this_panel
if not panel or panel.IsBlank():
Collection = None
Key = None
else:
Collection = self.Collection
Key = panel.Key
IsInfoPassingBackInvalid = self.CheckBeforeClose(Key)
if IsInfoPassingBackInvalid:
msg = "The information passed by this form to the parent form is not the correct type. If you continue closing with form, the record will be saved. However, it will not be passed back to the parent. Do you wish to continue?"
ans = dlgSimpleMsg('ERROR', msg, wx.YES_NO, wx.ICON_ERROR, Parent = self)
if ans == wx.ID_NO :
return
else:
Collection = None
Key = None
CloseForm(self, refresh=True, collection=Collection, key=Key)
return
# if current control is Combo, now open the popup window
if type(this_ctrl) in [ Combo, wx._controls.ComboBox ] and \
evtType == wx.wxEVT_LEFT_UP:
this_ctrl.OpenPopup()
# process control specific event
if evtType in SUPPORTED_COMMANDS:
# if there are nothing in the scroll window and action is not ADD
if panels == [] and menuId not in [wx.ID_ADD, wx.ID_OPEN]:
dlgSimpleMsg('ERROR', "Fields are not defined in this form. Please add a field first", wx.OK, wx.ICON_ERROR, Parent = self)
return
# First check pre-defined actions
if menuId == wx.ID_FIND : # open search dialog
row_panel = self.GetFocusedPanel()
dlgFind(self.pn_view, row_panel.GetDefaultFocus(), self, -1, "").Show()
elif menuId in [wx.ID_ADD, ID_MENU_COPY_RECORD, ID_SPECIAL_RECORD_ADD] : # ADD a new row panel
if Exist(prev_panel) and prev_panel.Id == 0 : # if there is new panel which is not saved
dlgSimpleMsg('GUI ERROR', "Can't add new panel. Please enter data first", wx.OK, wx.ICON_ERROR, Parent = self)
prev_ctrl.SetFocus()
else:
if menuId == ID_MENU_COPY_RECORD:
# Get the data by copying the current record
frm = prev_panel.GetTopLevelParent()
Collection = getattr(DB, frm.Collection)
# Note that this actually changes the DB
RecordData = Collection.Copy(prev_panel.Key)
self.AddPanel(data=RecordData)
elif menuId == ID_SPECIAL_RECORD_ADD:
# Get the data by calling the frame window function
# with the current record key
frm = prev_panel.GetTopLevelParent()
if hasattr(frm,'SpecialRecordAdd'):
RecordData = frm.SpecialRecordAdd(prev_panel)
self.AddPanel(data=RecordData)
else:
# No data for a new record
self.AddPanel()
elif menuId == wx.ID_DELETE : # Delete row panel and record in database
self.DeletePanel(this_panel)
return
else: # Then, look for control specific actions
# handle the case of the sort sub menu
if evtType == wx.wxEVT_COMMAND_MENU_SELECTED:
EventID = ID_EVT_SORT
btn = self.FindWindowById(menuId)
else:
btn = this_panel.FindWindowById(menuId)
if type(btn) == ListCtrlComboPopup : # if current control is list control in popup window
btn = btn.GetParent().GetParent() # find Combo Control to retrieve event action id
EventID = btn.evtID
if EventID == ID_EVT_SORT: # sort
SortPanels( self.pn_view, btn)
# If there is a panel which is focused or new scroll window to the panel
cur_panel = self.FindRowPanelByStatus([wx.ID_OPEN, wx.ID_NEW])
if cur_panel:
scrollPanel(self.pn_view, cur_panel)
elif EventID == ID_EVT_OWN:
btn.evtData(evt) # call own event handler
except:
# If an exception occur, display error message and move the focus to previous control
if evtType == wx.wxEVT_CLOSE_WINDOW :
ans = dlgErrorMsg(yesno=True, Parent = self)
if ans == wx.ID_YES : return
CloseForm(self, False)
return
else:
dlgErrorMsg(Parent = self)
if Exist(this_panel) and this_panel.isRow:
this_panel.SetStatus(wx.ID_NONE)
if type(this_ctrl) == Checkbox:
this_ctrl.SetValue(not this_ctrl.GetValue())
if Exist(prev_panel) :
prev_panel.SetStatus(prev_panel.Id, False)
prev_ctrl.SetFocus()
self.SetPrevTarget(prev_panel, prev_ctrl)
self.SetCurrTarget(prev_panel, prev_ctrl)
# If the event is activated by button clicking or menu selection,
# the event should not be propagated.
# For the event propagation, refer wxPython manual
if evtType not in SUPPORTED_COMMANDS:
evt.Skip()
def CheckBeforeClose(self, Key):
""" Dummy method to be overridden when inherited """
# This method should raise an error if closing the form will pass
# an invalid type record to the previous form. This function serves
# as default and does nothing and should be overridden by a derived
# class to take effect of this feature.
# The function returns False if problem is encountered. A True should
# Trigger a message to the user.
return False
# delete panel and record by clicking 'x' button or selecting 'delete' menu
def DeletePanel(self, this_panel=None):
""" Method to delete row panel and object in database"""
# If target panel is not set, check current target
if this_panel == None:
this_panel = self.GetCurrTarget('obj', 'panel')
# If can't find target panel, display error message
if self.GetCurrTarget('id', 'panel') < 0:
dlgSimpleMsg('GUI ERROR', "Please select a panel", wx.OK, wx.ICON_ERROR, Parent = self)
return
# If target panel is in database or data was entered for new row panel
# check user's response
if this_panel.Id > 0 or not this_panel.isEmpty:
RecordName = this_panel.TextRecordID()
msg = "Do you really want to delete this record? \n"
msg += "The " + RecordName + " will be deleted permanently"
answer = dlgSimpleMsg('WARNING', msg, wx.YES_NO, wx.ICON_QUESTION, Parent = self)
if (answer == wx.ID_NO): return
# delete record in database
if this_panel.Id > 0:
CollectionInstance = getattr(DB, self.Collection)
CollectionInstance.Delete(this_panel.Key, ProjectBypassID = self.idPrj)
sizer = this_panel.GetContainingSizer() # get sizer that contains this panel
sizer.Detach(this_panel) # detach this panel from containing sizer
wx.CallAfter(this_panel.Destroy) # remove panel
self.Layout() # re-arrange panels
# set the status of the previous panel as NONE --> delete arrow mark
prev_panel = self.GetPrevTarget('obj', 'panel')
if Exist(prev_panel):
prev_panel.SetStatus(wx.ID_NONE)
self.ResetTargetInfo() # initialize target(panel, control) ids
if len(self.pn_view.GetChildren()) == 0: return # if all row panel have been deleted, return
adjustScrollBar(self, self.pn_view) # change the length of the scroll bar
self.SetDefaultTarget(0) # set current target
# Find new panel
def FindRowPanelById(self, pid):
"""Find RowPanel object in a frame and return it using Panel Id"""
for panel in self.pn_view.GetChildren():
if panel.Id == pid : return panel
return None
# Find new panel
def FindRowPanelByStatus(self, status):
"""Find RowPanel object in a frame and return it using Panel Status"""
if type(status) not in [list, tuple]:
status = [status]
for panel in self.pn_view.GetChildren():
if panel.Status in status :
return panel
return None
def GetBlankPanel(self):
""" Method to fine newly added row panel """
return self.FindWindowById(0)
def ReturnDisplayedRecordKeys(self):
"""Return all the record ID's in the form"""
Result = []
# Note that a new record will not have a Key parameter
for panel in self.pn_view.GetChildren():
if 'Key' in dir(panel):
Result.append(panel.Key)
return Result
# retrieve current target information : id/object of panel/control
def GetCurrTarget(self, type='obj', what='both'):
""" Mehtod to retrieve current targets(panel/control) information(id or object)"""
if type == 'id':
if what == 'panel':
return self._id_this_panel
elif what == 'ctrl':
return self._id_this_ctrl
else:
return self._id_this_panel, self._id_this_ctrl
elif type == 'obj':
if what == 'panel':
return self.FindRowPanelById(self._id_this_panel)
elif what == 'ctrl':
return self.FindWindowById(self._id_this_ctrl)
else:
return self.FindRowPanelById(self._id_this_panel), \
self.FindWindowById(self._id_this_ctrl)
def GetFocusedPanel(self):
""" Find a panel whose status is OPEN or NEW """
panels = self.pn_view.GetChildren()
status = [ panel.Status for panel in panels ]
id_panel = [ panel.Id for panel in panels ]
if wx.ID_OPEN in status:
id = id_panel[status.index(wx.ID_OPEN)]
return self.FindRowPanelById(id)
elif wx.ID_NEW in status:
id = id_panel[status.index(wx.ID_NEW)]
return self.FindRowPanelById(id)