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中的属性 代码中 说明 ...
随机推荐
- Android -- 获取接口数据的三个方法
1. compile 'com.loopj.android:android-async-http:1.4.9': AsyncHttpClient client = new AsyncHttpCli ...
- JFinal 项目 在tomcat下部署
原文:http://my.oschina.net/jfinal/blog/353062 首先明确一下 JFinal 项目是标准的 java web 项目,其部署方式与普通 java web 项目没有任 ...
- ActiveMQ入门实例Demo
前面我们已经搭建和配置好了ActiveMQ,下面来看一个Demo,体验一下MQ. JMS 消息模型 JMS消息服务应用程序结构支持两种模型:点对点模型,发布者/订阅者模型. (1)点对点模型(Queu ...
- MySQL练习题
MySQL练习题 一.表关系 请创建如下表,并创建相关约束 二.操作表 1.自行创建测试数据 2.查询“生物”课程比“物理”课程成绩高的所有学生的学号: 3.查询平均成绩大于60分的同学的学号和平均成 ...
- (转)git常见错误
如果输入$ Git remote add origin git@github.com:djqiang(github帐号名)/gitdemo(项目名).git 提示出错信息:fatal: remote ...
- Android源码——Activity组件的启动过程
根Activity启动过程 Launcher启动MainActivity的过程主要分为6个步骤: 一.Launcher向ActivityManagerService发送一个启动MainActivity ...
- JS控制,返回上一页之后强行刷新一次
网站建设过程中,提交页面后我们经常要用到window.history.go(-1)返回上一页,因为页面的缓存功能,我们只能返回上次操作的页面,但在删除等操作中,我们希望实时看到删除项目后的页面,这就要 ...
- java swing 双人五子棋源代码
import java.awt.Color; import java.awt.Font; import java.awt.Graphics; import java.awt.Toolkit; impo ...
- 团队作业week16
0.新成员如何管理和使用这个工程? 对于后端而言,因为我们也是初次接触Django,因此从我们学习到程序编写过程中用到的各种知识都有学习笔记,比如Django的详细使用,Django数据库的使用.跨域 ...
- 【gulp】工作中的实战
写这篇文章的目的是为了以后的项目中懒得再去配gulp,直接可以拿这篇博客中的来用,因为有时候配置还是挺烦人的. gulp相关插件的介绍 用法比较简单,假设大家都会用gulp,下面主要介绍一下一些插件的 ...