这一篇教程,我们来完成在线文件编辑这个练习项目的第二阶段。

我们在上一篇教程中,已经了解了CGI的使用,如果大家有兴趣的话,可以尝试实现一个在线留言的功能。

类似下图所示:

python多人交互游戏开发中期报告下一步计划 python多人协作在线编辑文档_html

注意:这并不是在线聊天的功能,页面不经过刷新是看不到最新留言的。

上方的留言功能允许用户输入昵称,默认显示游客。当留言成功时,新的留言在最上方出现。

此案例源代码在本节资料中获取,下载地址在本文末尾。

接下来,我们回归正题。

第二阶段的内容包括三个文件:index.html、edit.py和save.py。

从文件名上大家就能够理解这三个文件各自的功能。

“index.html”这个文件,是我们要打开的首页。

实际上这么命名是有讲究的,因为在访问Web站点时,如果没有指定访问哪个页面(例如:http://219.142.209.7:8888/),会自动打开站点设置的默认页面,通常情况下默认页面就是“index.html”这个文件。在IIS信息服务管理器中,点击站点名称就能够在窗口中部看到默认页面的设置项。

而edit.py和save.py这两个文件一个是编辑功能的实现,一个是保存功能的实现。

下面,我先通过截图让大家了解要实现的目标。

index页面:

python多人交互游戏开发中期报告下一步计划 python多人协作在线编辑文档_文件名_02

edit页面:

python多人交互游戏开发中期报告下一步计划 python多人协作在线编辑文档_文件名_03

save页面:

python多人交互游戏开发中期报告下一步计划 python多人协作在线编辑文档_示例代码_04

有了清晰的目标,才能更好地理解每一段示例代码。

1、创建“index.html”文件

在PyCharm中新建一个HTML文件。

python多人交互游戏开发中期报告下一步计划 python多人协作在线编辑文档_html_05

在创建好的文件中会自带一些代码。

这些代码中有一些需要修改,包括:语言类型、字符集和页面标题。

示例代码:

<!DOCTYPE html>
<html lang="en"> <!"en"需要改为"zh_cn">
<head>
    <meta charset="UTF-8"> <!"UTF-8"需要改为"gbk">
    <title>Title</title> <!"Title"需要改为自定义的页面标题>
</head>
<body>

</body>
</html>

按照上方的注释修改完之后,我们需要添加一些表单标签的代码。

示例代码:

<!DOCTYPE html>
<html lang="zh_cn"> <!修改后的代码>
<head>
    <meta charset="gbk"> <!修改后的代码>
    <title>在线文本编辑</title><!haha> <!修改后的代码>
</head>
<body>
<form action="edit.py" method="post"> <!新增加的代码>
    <b>请输入文件名称:</b><br/>
    <input type="text" name="filename">
    <input type="submit" value="打开">
</form>
</body>
</html>

新增加的代码是一对<form>标签和其中的内容。

在<form>标签中,我们需要指定属性“action”为“edit.py”,也就是说当这个表单提交时,由“edit.py”这个脚本进行处理。而后面一个属性“method”需要指定数据的提交的方式,默认为“post”,也可以使用“get”。

在两个<form>标签之间,有两个<input>标签,一个是文本框,一个是提交按钮。因为要获取文本框中输入的内容,所以文本框标签中要设置“name”属性,这个属性的值可以自定义,例如这里填入的是“filename”。

2、创建“html.py”文件

前面没有提到创建这个文件。

这个文件作用是什么呢?

我们在上一篇教程中了解到,在CGI脚本中,我们必须要添加指定的语句。

示例代码:

print("Content-Type: text/html")
print("")

或者


print("Content-Type: text/html\n")


上述代码是输出头部信息,如果不添加的话,页面会报错无法访问。

但是如果每一个CGI脚本文件都添加的话感觉很麻烦。

所以,我们可以单独创建一个“html.py”文件(必须用这个名称),把上述代码写入。

这样的话,CGI脚本中不添加上述代码也能够正常打开。

注意:此方法未查到相关文档依据,具体原因不明,是小楼无意中的一个发现。如有高手知晓,敬请留言解惑,谢谢!

3、创建“edit.py”文件

“edit.py”文件用于处理“index.html”页面提交的数据。

实际上,是要通过“index.html”页面中输入的文件名打开本地相应的文件。

那么,我们就需要在代码中,对“index.html”页面中文本框的输入进行判断,如果没有输入文件名或没有找到相应的文件都要给出提示,并给出一个能够点击返回的文本链接。

类似下图:

python多人交互游戏开发中期报告下一步计划 python多人协作在线编辑文档_html_06

示例代码:

# -*- coding:gbk -*-
import cgi,sys
'''
想要学习Python?Python学习交流群:984632579满足你的需求,资料都已经上传群文件,可以自行下载!
'''
form = cgi.FieldStorage()
filename = form.getvalue('filename')
if not filename:
    print('<font color="red">请输入一个文件名称!</font>')
    print('<a href="\index.html" >返回</a>')
    sys.exit()
try:
    text = open(filename).read()
except FileNotFoundError:
    print('<font color="red">文件未找到,请输入正确的文件名称!</font>')
    print('<a href="\index.html" >返回</a>')
    sys.exit()

在上方代码中,要注意以下几点:

  • 声明编码为“gbk”。
  • 导入sys模块,通过sys.exit()方法结束当前语句下方代码的执行。
  • 通过try/except语句,捕获文件不存在的异常(FileNotFoundError),进行相应处理。
  • 通过<font>标签并设置“color”属性,创建红色文字提示。
  • 通过<a>标签添加返回链接。

如果文件名正确输入,能够打开本地的某个文件,这时我们让CGI脚本将正确的HTML页面代码输出给浏览器。

示例代码:

print('''
<!DOCTYPE html>
<html lang="zh_cn">
<head>
    <meta charset="gbk">
    <title>编辑</title>
</head>
<body>
<form action="save.py" method="post">
    <input type="hidden" value="%s" name="filename"/>
    <b>内容编辑:</b>%s<br/>
    <in>
    <textarea rows="20" cols="60" name="content">%s</textarea><br/>
    <b>输入密码:</b>
    <input type="password" name="password"/>
    <input type="submit" value="保存文档">
</form>
</body>
</html>
''' % (filename,filename,text))

在上方代码中,要注意以下3点:

  • <form>标签包含的表单内容,要通过”save.py”脚本文件进行处理;
  • 从上往下,第一个<input>标签的“type”属性值为“hidden”,是将这个标签隐藏,不在页面中显示;而它的“value”属性值为“%s”,写入的是打开的本地文件的名称。这个标签起到的作用是,将当前打开的文件名称传递给”save.py”脚本文件进行后续保存文件内容的处理。
  • “type”属性为“password”的<input>标签是一个密码类型的文本框,用于输入密码。

4、创建“save.py”文件

保存文件的处理比较简单,主要是一些验证的处理。

首先,验证文件名称、编辑内容和密码是否都正常被获取。

示例代码:

import cgi, sys, hashlib

form = cgi.FieldStorage()
filename = form.getvalue('filename')
password = form.getvalue('password')
content = form.getvalue('content')

if not (filename and password and content):  # 如果有任何一项内容为空值
    print('<font color="red">无效的输入!</font>')
    print('<a href="\edit.py" >返回</a>')
    sys.exit()

然后,还要验证输入的密码和预留密码是否一致。

示例代码:

pwd = hashlib.md5()  # 创建md5加密对象
pwd.update(password.encode())  # 添加加密内容
if pwd.hexdigest() != 'e10adc3949ba59abbe56e057f20f883e':  # 将加密后的md5码与预置md5码对比
    print('<font color="red">无效的密码!</font>')
    print('<a href="\edit.py" >返回</a>')
    sys.exit()

最后,如果以上验证都没有问题,保存文件,给出保存成功的提示。

示例代码:

with open(filename, 'w') as file:
    file.write(content)
print('<font color="green">文档已保存!</font>')
print('<a href="\index.html" >返回</a>')

到这里,我们就完成了整个练习项目。