设为首页 收藏本站
查看: 1745|回复: 0

[经验分享] python gui之tkinter界面设计pythonic设计

[复制链接]
累计签到:2 天
连续签到:1 天
发表于 2015-12-2 08:49:13 | 显示全部楼层 |阅读模式
  ui的设计,控件id的记录是一件比较繁琐的事情。
  此外,赋值和读取数据也比较繁琐,非常不pythonic。
  有没有神马办法优雅一点呢?life is short。
  
  鉴于控件有name属性,通过dir(Entry_obj)得知,存放在一个_name的属性里面。于是就有了以下代码:



Entry(frame,name='your_id1').grid(row=x1,column=y1)
Entry(frame,name='your_id2').grid(row=x2,column=y2)
...
Entry(frame, name='your_idn').grid(row=xn,column=yn)

  当然还有其他输入控件,此处略。
  可以通过 frame.grid_slaves()得到其上面所有的控件。
  然后可以通过 __class__来判断具体类型。可以在此基础上判断哪些控件是输入控件,如下:



for ctrl in frame.grid_slaves():
if ctrl.__class__.__name__ in ('Entry','Text',....):
return True
return False

  
  此外,我们还需要一个映射表,来关联界面控件和我们的模型。
  mapper={'model_attr1':ctrol_id1, 'model_attr2':ctrol_id2,...., 'model_attrn':ctrol_idn}
  
  通过这个映射表,我们可以方便的进行表单数据的设置和读取,假设所有输入控件皆为Entry,于是就有了一下读取表单的代码:



    def get_form_data(self):
vals = {}.fromkeys(self.mapper.keys(),False)
ctrls = dict([(x._name,x) for x in self.input_ctrls])
for k,v in self.mapper.items():
ctrl = ctrls.get(v,False)
if ctrl and ctrl.get():
vals.update({k:ctrl.get()})
#print vals
logging.debug('product form data:%s'%vals)
return vals

  
  关于input_ctrls的说明,因为控件的从属关系,本质上应该是一个树。
  可以采用一个设计:
  将控件放入到frame中,所有的frame放入到一个列表中frame_list。
  frame中的控件用grid方式布局。于是就有了以下读取界面上所有输入控件的代码:



    def _is_input(self, ctrl_obj):
"""
是否为输入控件
:param ctrl_obj:
:return:
"""
if ctrl_obj:
name = ctrl_obj._name
matches= ['txt_','cb_']
if (name.split('_')[0]+'_') in matches:
return True
return False
def _get_input_ctrls(self, container):
if container:
cs = container.grid_slaves()
cs.extend(container.pack_slaves())
for c in cs:
if self._is_input(c):
self.input_ctrls.append(c)
else:
self._get_input_ctrls(c)
def get_input_ctrls(self):
self.input_ctrls=[]
for f in self.frames:
self._get_input_ctrls(f)
return self.input_ctrls

  是否输入控件采用了根据_name属性进行的判断。此处需要有coding规范约束。
  实际上可以改写_is_input逻辑,根据__class__来进行枚举判断。
  
  相应的设置表单的值逻辑如下:



    def set_data(self, data):
if not isinstance(data,dict):
raise TypeError
if not self.mapper:
raise "set_mapper method must be called before this method being called."
ctrls = dict([(x._name,x) for x in self.get_input_ctrls()])
for k,v in self.mapper.items():
if True:#data.get(k,None) is not None:
ctrl_obj = ctrls.get(v)
if ctrl_obj:
state = ctrl_obj['state']
ctrl_obj.configure(state='normal')
if isinstance(ctrl_obj,NewCBtn):
ctrl_obj.checked(data.get(k))
elif isinstance(ctrl_obj,NewCBox):
vals = data.get(k)
if isinstance(vals,(list,tuple)):
val= vals[-1]
if not ctrl_obj.dictionary:
t={vals[-1]:vals[0]}
ctrl_obj.set_dict(t)
ctrl_obj.set(val)
else:
ctrl_obj.value(vals)
elif isinstance(ctrl_obj,NewImage):
ctrl_obj.Image(data.get(k))
elif isinstance(ctrl_obj,Text):
ctrl_obj.delete("1.0",END)
ctrl_obj.insert("1.0",data.get(k))
elif isinstance(ctrl_obj,Entry):
ctrl_obj.delete(0,END)
ctrl_obj.insert(0, str(data.get(k) or ''))
else:
logging.warning('unkown control type object:%s'%ctrl_obj.__class__.__name__)
ctrl_obj.configure(state=state)
  
  其中,类Newxxx为自己定义的类:

  NewCBox



class NewCBox(ttk.Combobox):
def __init__(self, master, dictionary={}, *args, **kw):
if not kw:kw={'values':sorted(list(dictionary.keys()))}
else:kw.update({'values':sorted(list(dictionary.keys()))})
ttk.Combobox.__init__(self, master, *args,  **kw)
self.dictionary = dictionary
#self.bind('<<ComboboxSelected>>', self.selected) #purely for testing purposes
self.bind('<Control-a>', self.select_all_input)
def select_all_input(self,event):
#选择输入文本,不起作用
#print 'Ctrl+A'
text = self.get()
vals = self.cget('values')
for i in range(len(vals)):
if vals==text:
self.current(i)
def set_dict(self, dictionary):
self.dictionary=dictionary
self['values']= sorted(list(dictionary.keys()))
def value(self,new_val=None):
if new_val is None:
key = self.get()
d= self.dictionary
k1=key.encode('UTF-8')#ASCII码就是UTF-8编码的最少字节的版本
k2=key.encode('GB2312')
k3=key.encode('CP936')
return d.get(key,d.get(k1,d.get(k2,d.get(k3,False))))else:
for k,v in self.dictionary.items():
if str(new_val)==str(v):
self.set(k)
return k
# def selected(self, event): #Just to test
#     print(self.value())
  NewImage



import io
class NewImage(Label):
def __init__(self, master=None,image_path=r'nullImage.png', cnf={}, **kw):
image = Image.open(image_path)
self.default_image = ImageTk.PhotoImage(image)
self.current_image = self.default_image
Label.__init__(self, image=self.current_image, master=master, cnf=cnf,**kw)
def Image(self,base64_source=None,encoding='base64'):
if not base64_source is None:
if base64_source:
base64_source = resize_img(base64_source,size=(100,100))
image_stream = io.BytesIO(base64_source.decode(encoding))
image = Image.open(image_stream)
self.current_image =ImageTk.PhotoImage(image)#PhotoImage(file='pro.gif')# PhotoImage(data=background_stream.getvalue())
self['image']=self.current_image
else:
self['image']=self.default_image
return self['image']
  
  NewCBtn



class NewCBtn(Checkbutton):
def __init__(self, master,*args, **kw):
self.value = IntVar()
if not kw:kw={'variable':self.value}
else:kw.update({'variable':self.value})
Checkbutton.__init__(self,master,*args,**kw)
def checked(self,value=None):
if value is not None:
self.value.set(value and 1 or 0)
return self.value.get()
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-146104-1-1.html 上篇帖子: python 查找文件内容 下篇帖子: Python 的各种符号
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表