使用Python和Tkinter构建简单的Markdown编辑器(2)

在这里,我们创建了一个宽度为1的Text小部件。不要误会,以为错了-这里的大小是使用比例来完成的。 当我们将其放入输出框中时,您将在接下来的几秒钟内更清楚地了解它。

然后,我们将其包装到框架中,并使其在水平和垂直方向上均可拉伸。

运行脚本时,您会看到“Multiline Input Field”已接管了整个“窗口”。 如果您开始写它,您可能会注意到字符太小了。

使用Python和Tkinter构建简单的Markdown编辑器

我已经知道会出现这个问题。 这就是为什么我之前告诉过您创建自定义字体对象(self.myfont)的原因。 现在,如果您执行以下操作:

self.inputeditor = Text(self, , font=self.myfont)

(这里,我们告诉Text小部件使用自定义字体,而不是默认的小字体!)

...输入字段的字体大小将增加到15。运行脚本以检查是否一切正常。

使用Python和Tkinter构建简单的Markdown编辑器

现在,我认为是时候添加outputbox了,我们在编写时将看到markdown源代码的html输出。

为此,我们要添加一个HTMLLabel,在init_window函数中是这样的:

self.outputbox = HTMLLabel(self,, background="white", html="<h1>linuxidc.com</h1>")
self.outputbox.pack(fill=BOTH, expand=1, side=RIGHT)
self.outputbox.fit_height()

我们使用tkhtmlview中的HTMLLabel,宽度仍旧为1。 我们将宽度设置为1,因为窗口将在输入字段和输出框之间以1:1的比例共享(运行脚本时您会明白我的意思)。

html关键字参数存储将在第一次显示的值。

然后,将其打包在窗口中,将side作为RIGHT置于输入字段的右侧。fit_height()使文本适合小部件。

现在运行代码,如下所示:

使用Python和Tkinter构建简单的Markdown编辑器

 

现在,如果您开始在输入字段中书写,输入时输出不会得到更新。 那是因为我们还没有告诉我们的程序这样做。

为此,我们首先要与编辑器绑定一个事件。 然后,你进行修改文本,输出都会得到更新,如下所示:

self.inputeditor.bind("<<Modified>>", self.onInputChange)

将这一行放到init_window()函数中。

这一行告诉inputeditor在文本改变时调用onInputChange函数。但是因为我们还没有那个函数,我们需要把它写出来。

...
def onInputChange(self , event):
    self.inputeditor.edit_modified(0)
    md2html = Markdown()
    self.outputbox.set_html(md2html.convert(self.inputeditor.get("1.0" , END)))

在第一行中,我们使用edit_modified(0)重置修改后的标志,以便重用它。否则,在第一次事件调用之后,它将不再工作。

接下来,我们创建一个名为md2html的Markdown对象。最后一行(上面标红那行),首先我们…等等!最后一行可能会让一些读者感到困惑。我把它分成三行。

markdownText = self.inputeditor.get("1.0" , END)
html = md2html.convert(markdownText)
self.outputbox.set_html(html)

在第一行中,我们从输入字段的顶部到底部获取markdown文本。第一个参数,self.inputeditor.get,告诉它从第一行的第0个字符开始扫描(1.0 => [LINE_NUMBER].[CHARACTER_NUMBER]),最后一个参数告诉它在到达末尾时停止扫描。

然后,我们使用md2html.convert()函数将扫描的markdown文本转换为html,并将其存储在html变量中。

最后,我们告诉outputbox使用.set_html()函数来显示输出!

运行脚本。您将看到一个功能几乎正常的markdown编辑器。当您输入输入字段时,输出也将被更新。

但是…我们的工作还没有完成。用户至少需要能够打开和保存他们的文本。

为此,我们要在菜单栏中添加一个文件菜单。在这里,用户可以打开和保存文件,也可以退出应用程序。

在init_window函数中,我们将添加以下行:

self.mainmenu = Menu(self)
self.filemenu = Menu(self.mainmenu)
self.filemenu.add_command(label="打开", command=self.openfile)
self.filemenu.add_command(label="另存为", command=self.savefile)
self.filemenu.add_separator()
self.filemenu.add_command(label="退出", command=self.quit)
self.mainmenu.add_cascade(label="文件", menu=self.filemenu)
self.master.config(menu=self.mainmenu)

简单说一下:

在这里,我们定义了一个新菜单,框架作为它的父菜单。

接下来,我们定义另一个菜单和上一个菜单作为其父菜单。它将作为我们的文件菜单。

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/aa060f11093ffc8e46068cf79c3179e2.html