wxPython 自动提示文本框
1、原版和例子都在这里
在浏览器的地址栏,或者在百度、google 输入文字的时候,输入框的下面会把有关的项目都提示出来。
wxPython 没有提供类似的控件,google 了一下,发现了一个,很好用。
下面是核心文件 autocomplete.py
# -*- coding: utf-8 -*-
__license__ = """Copyright (c) 2008-2010, Toni Ruža, All rights reserved.
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'AS IS'
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE."""
__author__ = u"Toni Ruža <gmr.gaf@gmail.com>"
__url__ = "http://bitbucket.org/raz/wxautocompletectrl"
import wx
class SuggestionsPopup(wx.Frame):
def __init__(self, parent):
wx.Frame.__init__(
self, parent,
style=wx.FRAME_NO_TASKBAR|wx.FRAME_FLOAT_ON_PARENT|wx.STAY_ON_TOP
)
self._suggestions = self._listbox(self)
self._suggestions.SetItemCount(0)
self._unformated_suggestions = None
class _listbox(wx.HtmlListBox):
items = None
def OnGetItem(self, n):
return self.items[n]
def SetSuggestions(self, suggestions, unformated_suggestions):
self._suggestions.items = suggestions
self._suggestions.SetItemCount(len(suggestions))
self._suggestions.SetSelection(0)
self._suggestions.Refresh()
self._unformated_suggestions = unformated_suggestions
def CursorUp(self):
selection = self._suggestions.GetSelection()
if selection > 0:
self._suggestions.SetSelection(selection - 1)
def CursorDown(self):
selection = self._suggestions.GetSelection()
last = self._suggestions.GetItemCount() - 1
if selection < last:
self._suggestions.SetSelection(selection + 1)
def CursorHome(self):
if self.IsShown():
self._suggestions.SetSelection(0)
def CursorEnd(self):
if self.IsShown():
self._suggestions.SetSelection(self._suggestions.GetItemCount() - 1)
def GetSelectedSuggestion(self):
return self._unformated_suggestions[self._suggestions.GetSelection()]
def GetSuggestion(self, n):
return self._unformated_suggestions[n]
class AutocompleteTextCtrl(wx.TextCtrl):
def __init__(
self, parent,
height=300, completer=None,
multiline=False, frequency=250
):
style = wx.TE_PROCESS_ENTER
if multiline:
style = style | wx.TE_MULTILINE
wx.TextCtrl.__init__(self, parent, style=style)
self.height = height
self.frequency = frequency
if completer:
self.SetCompleter(completer)
self.queued_popup = False
self.skip_event = False
def SetCompleter(self, completer):
"""
Initializes the autocompletion. The 'completer' has to be a function
with one argument (the current value of the control, ie. the query)
and it has to return two lists: formated (html) and unformated
suggestions.
"""
self.completer = completer
frame = self.Parent
while not isinstance(frame, wx.Frame):
frame = frame.Parent
self.popup = SuggestionsPopup(frame)
frame.Bind(wx.EVT_MOVE, self.OnMove)
self.Bind(wx.EVT_TEXT, self.OnTextUpdate)
self.Bind(wx.EVT_SIZE, self.OnSizeChange)
self.Bind(wx.EVT_KEY_DOWN, self.OnKeyDown)
self.popup._suggestions.Bind(wx.EVT_LEFT_DOWN, self.OnSuggestionClicked)
self.popup._suggestions.Bind(wx.EVT_KEY_DOWN, self.OnSuggestionKeyDown)
def AdjustPopupPosition(self):
self.popup.Position = self.ClientToScreen((0, self.Size.height)).Get()
def OnMove(self, event):
self.AdjustPopupPosition()
event.Skip()
def OnTextUpdate(self, event):
if self.skip_event:
self.skip_event = False
elif not self.queued_popup:
wx.CallLater(self.frequency, self.AutoComplete)
self.queued_popup = True
event.Skip()
def AutoComplete(self):
self.queued_popup = False
if self.Value != "":
formated, unformated = self.completer(self.Value)
if len(formated) > 0:
self.popup.SetSuggestions(formated, unformated)
self.AdjustPopupPosition()
self.Unbind(wx.EVT_KILL_FOCUS)
self.popup.ShowWithoutActivating()
self.SetFocus()
self.Bind(wx.EVT_KILL_FOCUS, self.OnKillFocus)
else:
self.popup.Hide()
else:
self.popup.Hide()
def OnSizeChange(self, event):
self.popup.Size = (self.Size[0], self.height)
event.Skip()
def OnKeyDown(self, event):
key = event.GetKeyCode()
if key == wx.WXK_UP:
self.popup.CursorUp()
return
elif key == wx.WXK_DOWN:
self.popup.CursorDown()
return
elif key in (wx.WXK_RETURN, wx.WXK_NUMPAD_ENTER) and self.popup.Shown:
self.skip_event = True
self.SetValue(self.popup.GetSelectedSuggestion())
self.SetInsertionPointEnd()
self.popup.Hide()
return
elif key == wx.WXK_HOME:
self.popup.CursorHome()
elif key == wx.WXK_END:
self.popup.CursorEnd()
elif event.ControlDown() and unichr(key).lower() == "a":
self.SelectAll()
elif key == wx.WXK_ESCAPE:
self.popup.Hide()
return
event.Skip()
def OnSuggestionClicked(self, event):
self.skip_event = True
n = self.popup._suggestions.HitTest(event.Position)
self.Value = self.popup.GetSuggestion(n)
self.SetInsertionPointEnd()
wx.CallAfter(self.SetFocus)
event.Skip()
def OnSuggestionKeyDown(self, event):
key = event.GetKeyCode()
if key in (wx.WXK_RETURN, wx.WXK_NUMPAD_ENTER):
self.skip_event = True
self.SetValue(self.popup.GetSelectedSuggestion())
self.SetInsertionPointEnd()
self.popup.Hide()
event.Skip()
def OnKillFocus(self, event):
if not self.popup.IsActive():
self.popup.Hide()
event.Skip()
下面是用法举例 test_autocomplete.py
# -*- coding: utf-8 -*-
__license__ = """Copyright (c) 2008-2010, Toni Ruža, All rights reserved.
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'AS IS'
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE."""
__author__ = u"Toni Ruža <gmr.gaf@gmail.com>"
__url__ = "http://bitbucket.org/raz/wxautocompletectrl"
import sys
import os
import random
import string
import wx
from autocomplete import AutocompleteTextCtrl
template = "%s<b><u>%s</b></u>%s"
def random_list_generator(query):
formatted, unformatted = list(), list()
if query:
for i in xrange(random.randint(0, 30)):
prefix = "".join(random.sample(string.letters, random.randint(0, 10)))
postfix = "".join(random.sample(string.letters, random.randint(0, 10)))
value = (prefix, query, postfix)
formatted.append(template % value)
unformatted.append("".join(value))
return formatted, unformatted
def list_completer(a_list):
def completer(query):
formatted, unformatted = list(), list()
if query:
unformatted = [item for item in a_list if query in item]
for item in unformatted:
s = item.find(query)
formatted.append(
template % (item[:s], query, item[s + len(query):])
)
return formatted, unformatted
return completer
def test():
some_files = [
name
for path in sys.path if os.path.isdir(path)
for name in os.listdir(path)
]
quotes = open("taglines.txt").read().split("%%")
app = wx.App(False)
app.TopWindow = frame = wx.Frame(None)
frame.Sizer = wx.FlexGridSizer(3, 2, 5, 5)
frame.Sizer.AddGrowableCol(1)
frame.Sizer.AddGrowableRow(2)
# A completer must return two lists of the same length based
# on the "query" (current value in the TextCtrl).
#
# The first list contains items to be shown in the popup window
# to the user. These items can use HTML formatting. The second list
# contains items that will be put in to the TextCtrl, usually the
# items from the first list striped of formating.
field1 = AutocompleteTextCtrl(frame, completer=random_list_generator)
field2 = AutocompleteTextCtrl(frame, completer=list_completer(some_files))
field3 = AutocompleteTextCtrl(
frame, completer=list_completer(quotes), multiline=True
)
frame.Sizer.Add(wx.StaticText(frame, label="Random strings"))
frame.Sizer.Add(field1, 0, wx.EXPAND)
frame.Sizer.Add(wx.StaticText(frame, label="Files in sys.path"))
frame.Sizer.Add(field2, 0, wx.EXPAND)
frame.Sizer.Add(wx.StaticText(frame, label="Famous quotes"))
frame.Sizer.Add(field3, 0, wx.EXPAND)
frame.Show()
app.MainLoop()
if __name__ == '__main__':
test()
2、为了配合 wxFormBuilder 做了一些修改
如果是配合 wxFormBuilder 使用,就要先在设计器里画一个 TextCtrl,然后再修改生成的代码,把原来的 TextCtrl 替换成 AutocompleteTextCtrl。这样做有些麻烦,每次小改界面都要记得重新替换代码。
如果 AutocompleteTextCtrl 不是从无到有的生成一个新的 TextCtrl 实例,而是能够绑定到已经存在的 TextCtrl 上 就可以避免反复替换代码了。比如,在 wxFormBuilder 生成的代码中,我们有了一个现成的控件 m_textCtrl1,如果接下来能够像 AutocompleteTextCtrl(m_textCtrl1) 这样包装一下使用就好了。原版并没有提供这样的功能,不过修改一下也很简单。下面是修改后的 AutocompleteTextCtrl 类。
class AutocompleteTextCtrl(object):
def __init__( self, textCtrl, completer=None, height=300, frequency=250 ):
"""
textCtrl用于指出已经存在的wx.TextCtrl实例,其他参数意义和原版一样。
"""
self.textCtrl = textCtrl
self.height = height
self.frequency = frequency
if completer:
self.SetCompleter(completer)
self.queued_popup = False
self.skip_event = False
def SetCompleter(self, completer):
"""
Initializes the autocompletion. The 'completer' has to be a function
with one argument (the current value of the control, ie. the query)
and it has to return two lists: formated (html) and unformated
suggestions.
"""
self.completer = completer
frame = self.textCtrl.Parent
while not isinstance(frame, wx.Frame):
frame = frame.Parent
self.popup = SuggestionsPopup(frame)
frame.Bind(wx.EVT_MOVE, self.OnMove)
self.textCtrl.Bind(wx.EVT_TEXT, self.OnTextUpdate)
self.textCtrl.Bind(wx.EVT_SIZE, self.OnSizeChange)
self.textCtrl.Bind(wx.EVT_KEY_DOWN, self.OnKeyDown)
self.popup._suggestions.Bind(wx.EVT_LEFT_DOWN, self.OnSuggestionClicked)
self.popup._suggestions.Bind(wx.EVT_KEY_DOWN, self.OnSuggestionKeyDown)
def AdjustPopupPosition(self):
self.popup.Position = self.textCtrl.ClientToScreen((0, self.textCtrl.Size.height)).Get()
def OnMove(self, event):
self.AdjustPopupPosition()
event.Skip()
def OnTextUpdate(self, event):
if self.skip_event:
self.skip_event = False
elif not self.queued_popup:
wx.CallLater(self.frequency, self.AutoComplete)
self.queued_popup = True
event.Skip()
def AutoComplete(self):
self.queued_popup = False
if self.textCtrl.Value != "":
formated, unformated = self.completer(self.textCtrl.Value)
if len(formated) > 0:
self.popup.SetSuggestions(formated, unformated)
self.AdjustPopupPosition()
self.textCtrl.Unbind(wx.EVT_KILL_FOCUS)
self.popup.ShowWithoutActivating()
self.textCtrl.SetFocus()
self.textCtrl.Bind(wx.EVT_KILL_FOCUS, self.OnKillFocus)
else:
self.popup.Hide()
else:
self.popup.Hide()
def OnSizeChange(self, event):
self.popup.Size = (self.textCtrl.Size[0], self.height)
event.Skip()
def OnKeyDown(self, event):
key = event.GetKeyCode()
if key == wx.WXK_UP:
self.popup.CursorUp()
return
elif key == wx.WXK_DOWN:
self.popup.CursorDown()
return
elif key in (wx.WXK_RETURN, wx.WXK_NUMPAD_ENTER) and self.popup.Shown:
self.skip_event = True
self.textCtrl.SetValue(self.popup.GetSelectedSuggestion())
self.textCtrl.SetInsertionPointEnd()
self.popup.Hide()
return
elif key == wx.WXK_HOME:
self.popup.CursorHome()
elif key == wx.WXK_END:
self.popup.CursorEnd()
elif event.ControlDown() and unichr(key).lower() == "a":
self.SelectAll()
elif key == wx.WXK_ESCAPE:
self.popup.Hide()
return
event.Skip()
def OnSuggestionClicked(self, event):
self.skip_event = True
n = self.popup._suggestions.HitTest(event.Position)
self.textCtrl.Value = self.popup.GetSuggestion(n)
self.textCtrl.SetInsertionPointEnd()
wx.CallAfter(self.textCtrl.SetFocus)
event.Skip()
def OnSuggestionKeyDown(self, event):
key = event.GetKeyCode()
if key in (wx.WXK_RETURN, wx.WXK_NUMPAD_ENTER):
self.skip_event = True
self.textCtrl.SetValue(self.popup.GetSelectedSuggestion())
self.textCtrl.SetInsertionPointEnd()
self.popup.Hide()
event.Skip()
def OnKillFocus(self, event):
if not self.popup.IsActive():
self.popup.Hide()
event.Skip()
wxPython 自动提示文本框的更多相关文章
- Android 自学之自动完成文本框 AutoCompleteTextView
自动完成文本框(AutoCompleteTextView)从EditText派生而出,实际上他也是一个编辑框,但他比普通的编辑框多了一个功能:当用户输入一定字符后,自动完成文本框会显示一个下拉菜单,供 ...
- Android开发10.2:UI组件AutoCompleteTextView(自动完成文本框)
概述 AutoCompleteTextVeiw(自动完成文本框)从 EditText派生而出 PS :EditText用法介绍 当用户输入一定字符后,自动完成自动完成文本框会显示 ...
- android脚步---自动完成文本框
自动完成文本框AutoCompleteTextView,当用户输入一定字符时,自动完成文本框会显示一个下拉菜单,供用户选择,设置一个Adapter,该Adapter中封装了AutoCompleteTe ...
- Android零基础入门第47节:自动完成文本框AutoCompleteTextView
原文:Android零基础入门第47节:自动完成文本框AutoCompleteTextView 上一期学习的Spinner的使用,掌握的怎么样?本期一起来学习AutoCompleteTextView的 ...
- AutoCompleteTextView自动完成文本框
AutoCompleteTextView是从EditText派生出来的,比普通编辑框多了一个功能,当用户输入一定字符后,自动完成文本框会显示一个下拉单,供用户选择,当选中一个后,被选中的内容会显示在文 ...
- JS实现动态提示文本框可输入剩余字数(类似发表微博数字提示)
一.实现效果: 为了更直观的体现用户在文本框输入文本时能看到自己输入了多少字,项目中需要通过判断提示文本框剩余可输入字数. html & JS: <div> <textare ...
- javascript 文字大小自动适应文本框 (文字大小自动调整)
javascript 文字大小自动适应文本框 (文字大小自动调整) TOC 思考 思考一:面积法 思考二:微调法 代码 在进行类似微博墙之类的展示页面中,经常会遇到这样的需求:在固定大小的区域放入字数 ...
- Android自己主动提示文本框(AutoCompleteTextView)
自己主动提示文本框(AutoCompleteTextView)能够加强用户体验,缩短用户的输入时间(百度的搜索框就是这个效果). 首先.在xml中定义AutoCompleteTextView控件: a ...
- Android 开发笔记___AutoComplateTextView__自动完成文本框
原理:EdtText结合监听器TextWatcher与下拉框spinner,一旦监控到EditText的文本发生变化,就自动弹出适配好的文字下拉内容. 属性以及设置方法: XML中的属性 代码中 说明 ...
随机推荐
- nth-of-type在选择class的时候需要注意的一个小问题
查了下w3和MDN的手册,没发现有这个说明,写篇随笔记下. 1..class:nth-of-type(n)在选择class的时候,如果在class前面插入x个同类型标签,n需要加上x <!DOC ...
- 字符串编辑距离(Levenshtein距离)算法
基本介绍 Levenshtein距离是一种计算两个字符串间的差异程度的字符串度量(string metric).我们可以认为Levenshtein距离就是从一个字符串修改到另一个字符串时,其中编辑单个 ...
- Favorites of top 10 rules for success
Dec. 31, 2015 Stayed up to last minute of 2015, 12:00am, watching a few of videos about top 10 rules ...
- kettle中变量的设置和使用介绍
有没有能统一管理一个参数,然后让所有的transformation和job都可以读到呢? 答案是有 1.首先,打开.kettle\kettle.properties(个人主机是:C:\Users\fo ...
- Django之Model操作
Django之Model操作 本节内容 字段 字段参数 元信息 多表关系及参数 ORM操作 1. 字段 字段列表 AutoField(Field) - int自增列,必须填入参数 primary_ke ...
- Codeforces Round #382(div 2)
A.= = B. 题意:给出n个数和n1和n2,从n个数中分别选出n1,n2个数来,得到n1个数和n2个数的平均值,求这两个平均值的最大和 分析:排个序从后面抽,注意先从末尾抽个数小的,再抽个数大的 ...
- MVC跨域CORS扩展
一般的基于浏览器跨域的主要解决方法有这么几种:1.JSONP 2.IFrame方式 3.通过flash实现 4.CORS跨域资源共享 ,这里我们主要关注的是在MVC里面的CORS ...
- 敏捷组织中PMO应遵循的准则
敏捷改变了人们的工作方式,不仅仅是开发部门,而且还包括其它的部门,例如HR.财务以及PMO等.在大多数组织中,PMO是一个控制体.它指导项目团队的规范.模板以及流程.目前,大多数的IT组织都敏捷化了. ...
- Android基础总结(四)
网络图片查看器 确定图片的网址 发送http请求 URL url = new URL(address); //获取连接对象,并没有建立连接 HttpURLConnection conn = (Http ...
- VS2015插件
这里记录一下,VS2015使用的插件和具体用法链接 Refactoring Essentils:代码重构分析 http://vsrefactoringessentials.com/ 可以使用 取消 R ...