美文网首页Linux学习|Gentoo/Arch/FreeBSDLinux我用 Linux
Vim不好用?学不会?用Python山寨一个

Vim不好用?学不会?用Python山寨一个

作者: DigiHacker | 来源:发表于2020-02-12 21:30 被阅读0次

    以Vim的操作之高效被称为编辑器之神,是绝对不夸张的。小编遇上大规模修改代码,一直靠Vim临时写个宏操作,一路确认下去几分钟就能搞定。换个编辑器就很难这样操作了。

    但vim终究是老了,界面弄的再漂亮,总要开个命令行才行。又不支持图片,处理有格式的文本更是力不从心。虽然后起之辈很多,但总归没有vim编辑模式下移动那么方便。

    用了PyQt里的QTextEditor后,发现模拟个Vim真的很方便。Vim与其它文本编辑器最大的区别是模式。通过Esc键切换到编辑模式,就可以快速修改文本。QTextEditor有只读模式,但是光标会消失。那只好自定义一个editing布尔类型变量来切换模式。

    首先定义一个QTextEditor的子类,然后在keyReleaseEvent方法里判断按下的键,如果是Esc, 就设置editing变量为真。

    def keyReleaseEvent(self, event):
        if event.key() == Qt.Key_Escape:
            self.editing = True
    

    当进入到编辑模式后,输入字母键就不应该有回应了,除非是hjkl这样的光标操作键。QTextEditor的textCursor属性就是用来操作光标的。向下移动光标的实现代码如下

        def moveDown(self, cursor=None):
            if cursor == None:
                cursor = self.textCursor()
            cursor.movePosition(QTextCursor.Up, QTextCursor.MoveAnchor, 1)
            self.setTextCursor(cursor)
            return cursor
    

    同理,只需要替换QTextCursor.Up为QTextCursor.Down, QTextCursor.Left, QTextCursor.Right 就可以实现hjkl键的功能

    def keyPressEvent(self, event):
     if self.editing:
      if event.key() == Qt.Key_K:
       self.moveUp()
      elif event.key() == Qt.Key_J:
       self.moveDown()
      elif event.key() == Qt.Key_H:
       self.moveLeft()
      elif event.key() == Qt.Key_L:
       self.moveRight()
      elif event.key() == Qt.Key_I:
       self.editing = False
      elif event.key() == Qt.Key_0:
       self.moveStartOfLine()
      elif event.key() == Qt.Key_Dollar:
       self.moveEndOfLine()
      elif event.key() == Qt.Key_A:
       self.moveEndOfLine()
       self.editing = False
      else:
       super().keyPressEvent(event)
    
     else:
      super().keyPressEvent(event)
    

    上面的代码还实现移动到行首,行尾以及添加和功能,一样可以用QTextCursor.StartOfLine, QTextCursor.EndOfLine 来实现。有的同学可能会奇怪为什么我们要用单独的方法,而不直接在keyPressEvent实现。这其实是一个简单的封装来隐藏Qt API。 如果将来要实现数字键加操作符或者解析vim脚本操作的话,大量调用Qt API就容易乱,不容易维护。而实现一套我们自己的API更好理解一些。

    最后我们来实现一个markdown的功能,并且实时渲染。 这是vim或者neovim绝对做不到的。

        def keyReleaseEvent(self, event):
            if event.key() == Qt.Key_Return:
                text = self.toPlainText().splitlines()
                last_line = text[-1]
                print(last_line)
                if last_line.startswith('#'):
                    self.moveUp()
                    self.deleteLine()
                    self.moveDown()
                    self.deleteLine()
                    self.textCursor().insertFragment(QTextDocumentFragment.fromHtml(f"<h1>{last_line}</h1>"))
                    self.appendPlainText("")
                    
        def deleteLine(self):
            self.moveStartOfLine()
            cursor = self.textCursor()
            cursor.select(QTextCursor.LineUnderCursor)
            cursor.removeSelectedText()
            cursor.deletePreviousChar()
            cursor.deletePreviousChar()
            self.setTextCursor(cursor)
    

    如果最后一行也就是最新输入的一行以#开头的话,就代表是markdown里的H1, 我们删除最后一行以及回车键产生的空白行,再加上一行html代码。同时增加一行(因为按了回车键)

    其实这样的代码跟vim脚本没有太大区别,当我们将来实现脚本解析引擎时,就可以按操作符替换为对应的方法名。

    相关文章

      网友评论

        本文标题:Vim不好用?学不会?用Python山寨一个

        本文链接:https://www.haomeiwen.com/subject/fvxhfhtx.html