Как отменить при использовании QScintilla setText? - PullRequest
0 голосов
/ 09 февраля 2019

Позвольте мне начать с публикации небольших вспомогательных функций, которые я буду использовать для формулирования своих вопросов:

import textwrap
import sys
from pathlib import Path

from PyQt5.Qsci import QsciScintilla
from PyQt5.Qt import *  # noqa


def set_style(sci):
    # Set default font
    sci.font = QFont()
    sci.font.setFamily('Consolas')
    sci.font.setFixedPitch(True)
    sci.font.setPointSize(8)
    sci.font.setBold(True)
    sci.setFont(sci.font)
    sci.setMarginsFont(sci.font)
    sci.setUtf8(True)

    # Set paper
    sci.setPaper(QColor(39, 40, 34))

    # Set margin defaults
    fontmetrics = QFontMetrics(sci.font)
    sci.setMarginsFont(sci.font)
    sci.setMarginWidth(0, fontmetrics.width("000") + 6)
    sci.setMarginLineNumbers(0, True)
    sci.setMarginsForegroundColor(QColor(128, 128, 128))
    sci.setMarginsBackgroundColor(QColor(39, 40, 34))
    sci.setMarginType(1, sci.SymbolMargin)
    sci.setMarginWidth(1, 12)

    # Set indentation defaults
    sci.setIndentationsUseTabs(False)
    sci.setIndentationWidth(4)
    sci.setBackspaceUnindents(True)
    sci.setIndentationGuides(True)
    sci.setFoldMarginColors(QColor(39, 40, 34), QColor(39, 40, 34))

    # Set caret defaults
    sci.setCaretForegroundColor(QColor(247, 247, 241))
    sci.setCaretWidth(2)

    # Set edge defaults
    sci.setEdgeColumn(80)
    sci.setEdgeColor(QColor(221, 221, 221))
    sci.setEdgeMode(sci.EdgeLine)

    # Set folding defaults (http://www.scintilla.org/ScintillaDoc.html#Folding)
    sci.setFolding(QsciScintilla.CircledFoldStyle)

    # Set wrapping
    sci.setWrapMode(sci.WrapNone)

    # Set selection color defaults
    sci.setSelectionBackgroundColor(QColor(61, 61, 52))
    sci.resetSelectionForegroundColor()

    # Set scrollwidth defaults
    sci.SendScintilla(QsciScintilla.SCI_SETSCROLLWIDTHTRACKING, 1)

    # Current line visible with special background color
    sci.setCaretLineBackgroundColor(QColor(255, 255, 224))

    # Set multiselection defaults
    sci.SendScintilla(QsciScintilla.SCI_SETMULTIPLESELECTION, True)
    sci.SendScintilla(QsciScintilla.SCI_SETMULTIPASTE, 1)
    sci.SendScintilla(QsciScintilla.SCI_SETADDITIONALSELECTIONTYPING, True)


def set_state1(sci):
    sci.clear_selections()
    base = "line{} state1"
    view.setText("\n".join([base.format(i) for i in range(10)]))
    for i in range(0, 10, 2):
        region = (len(base) * i, len(base) * (i + 1) - 1)
        if i == 0:
            view.set_selection(region)
        else:
            view.add_selection(region)


def set_state2(sci):
    base = "line{} state2"
    view.setText("\n".join([base.format(i) for i in range(10)]))
    for i in range(1, 10, 2):
        region = (len(base) * i, len(base) * (i + 1) - 1)
        if i == 1:
            view.set_selection(region)
        else:
            view.add_selection(region)


class Editor(QsciScintilla):

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        set_style(self)

    def clear_selections(self):
        sci = self
        sci.SendScintilla(sci.SCI_CLEARSELECTIONS)

    def set_selection(self, r):
        sci = self
        sci.SendScintilla(sci.SCI_SETSELECTION, r[1], r[0])

    def add_selection(self, r):
        sci = self
        sci.SendScintilla(sci.SCI_ADDSELECTION, r[1], r[0])

    def sel(self):
        sci = self
        regions = []

        for i in range(sci.SendScintilla(sci.SCI_GETSELECTIONS)):
            regions.append(
                sci.SendScintilla(sci.SCI_GETSELECTIONNSTART, i),
                sci.SendScintilla(sci.SCI_GETSELECTIONNEND, i)
            )

        return sorted(regions)

У меня есть пара вопросов:

Вопрос1)

if __name__ == '__main__':
    app = QApplication(sys.argv)

    view = Editor()
    set_state1(view)
    view.move(1000, 100)
    view.resize(800, 300)
    view.show()
    app.exec_()

Я получу это (вы можете увидеть вопрос на снимке ниже):

enter image description here

Вопрос 2)

if __name__ == '__main__':
    app = QApplication(sys.argv)

    view = Editor()
    set_state1(view)
    set_state2(view)
    view.move(1000, 100)
    view.resize(800, 300)
    view.show()
    app.exec_()

Как я могу изменить код, чтобы иметь возможность восстановить состояние1 при нажатии Ctrl + Z?

Прямо сейчас при использовании Ctrl + Z вы не сможете получить состояние1:

enter image description here

в основном из-за того, как setText ведет себя:

Заменяет весь текущий текст на текст.Обратите внимание, что история отмен / повторов очищается с помощью этой функции.

Я уже пробовал некоторые функции, опубликованные в документах отменить и повторить , но пока не получилось.

Например, одна из моих попыток - сначала выбрать весь текст, а затем использовать replaceSelectedText и, наконец, восстановить выборки из предыдущего состояния вручную, результат был ужасным (я не хочупрокрутка редактора портится при отмене / повторном выполнении) ... По сути, я хотел бы получить то же чувство, что и SublimeText.

Кстати, это небольшой минимальный пример, но в реальном случае янакапливать кучу операций без частой фиксации сцинтиллы ... вот почему мне интересно выяснить, как выполнить откат в предыдущее состояние при использовании отменяемого setText ... В противном случае я бы хотел избежать использования Scintillaтакие функции, как insertAt , replaceSelectedText или аналогичные ... так как я использую встроенные функции строки Python для изменения буфера вternally.

EDIT:

Я почти уверен, что beginUndoAction & endUndoAction не помогут мне ответить на вопрос2, но ... как насчет SCI_ADDUNDOACTION?Хотя документы довольно запутанные, хотя ...: /

1 Ответ

0 голосов
/ 10 февраля 2019

Вопрос 1 : последний добавленный выбор автоматически устанавливается как выбор Main.Чтобы удалить его, добавьте строку sci.SendScintilla(sci.SCI_SETMAINSELECTION, -1) в конце функции set_state1.

Вопрос 2 :

  • То, как вы описали это, сохраниввыбор с использованием replaceSelectedText, а затем с помощью setCursorPosition / reselecting all selections и setFirstVisibleLine для восстановления позиции прокрутки - один из способов.
  • Просмотр источника C ++ setTextфункция:
// Set the given text.
void QsciScintilla::setText(const QString &text)
{
    bool ro = ensureRW();

    SendScintilla(SCI_SETTEXT, ScintillaBytesConstData(textAsBytes(text)));
    SendScintilla(SCI_EMPTYUNDOBUFFER);

    setReadOnly(ro);
}

Вы можете попробовать установить текст, используя sci.SendScintilla(sci.SCI_SETTEXT, b"some text"), который не сбрасывает буфер отмены / повтора.

...