[Tkinter text] 可移动文字的text

效果

在这里插入图片描述

思路

既然要拖动,那就要按下,抬起和鼠标拖动3个事件。需要是否按下和文字的变量。
关于复制,需要键盘按下事件,判断keycode是否为17(space)。代码很简单,应该很容易读懂。

实现

import tkinter as tk
import sys
class TkContentMoveAndCopyableText(tk.Text):
    "取这个名字纯粹是用来恶心人的"
    def __init__(self,*a,**b):
        super().__init__(*a,**b)
        self.content = ""
        self.draging = False
        self.ctrl = False
        self.bind("<KeyPress>",self.ctrldown,1)
        self.bind("<KeyRelease>",self.ctrlup,1)
        self.bind("<B1-Motion>",self.handledrag,1)
        self.bind("<Button-1>",self.handledown,1)
        self.bind("<ButtonRelease-1>",self.handleup,1)
    def ctrlup(self,e):
        if e.keycode == 17:
            self.ctrl = False
    def ctrldown(self,e):
        if e.keycode == 17:
            self.ctrl=True
    def handledown(self,e):
        try:
            if self.insel(e):
                self.content = self.get("sel.first","sel.last")
                self.sel = (self.index("sel.first") , self.index("sel.last"))
                return "break"
        except:
            self.content = ""
    def handledrag(self,e):
        if self.content:
                self.draging = True
                self.mark_set("insert","@%d,%d"%(e.x,e.y))
                return "break"
    def handleup(self,e):
        if self.draging:
            if not self.ctrl:
                self.delete(*self.sel)
            self.insert("insert",self.content)
            self.draging = False
            self.content = ""
        else:
            if self.content:
                self.tag_remove("sel","1.0","end")
                self.content = ""
                self.mark_set("insert","@%d,%d"%(e.x,e.y))
    def insel(self,e):
        "其实可以用text.compare代替,判断insert标记是否在sel标签里"
        try:
            y1,x1 = map(int,self.index("sel.first"       ).split("."))
            y2,x2 = map(int,self.index("@%d,%d"%(e.x,e.y)).split("."))
            y3,x3 = map(int,self.index("sel.last"        ).split("."))
            if y3 > y2 > y1:
                return 1
            if y2 == y1 and x2 > x1:
                return 1
            elif y2 == y3 and x2 < x3:
                return 1
        except:
            pass
def test():
    root = tk.Tk()
    root.title("TextPlus")
    TextPlus().pack(expand=1,fill="both")
    root.mainloop()
if __name__ == "__main__":
    test()

用例

在这

总结

这个还是挺简单的,但比较实用,毕竟使用tk做的ide也有几个,向idle,thonny等,如果有人能做一个插件实现这种效果,就更好了!
本文发于CSDN于2022/2/6 14:33

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值