Gre*_*reg 4 python vba ms-word win32com
我正在尝试使用 Python 自动化 word 以替换 word 文档中的文本。(如果这很重要的话,我在 Word 2003 和 Python 2.4 上)
下面我的替换方法的第一部分适用于除文本框中的文本之外的所有内容。文本只是没有被选中。我注意到当我手动进入 Word 并按 ctrl-A 时,除文本框外的所有文本都被选中。
到目前为止,这是我的代码:
class Word:
def __init__(self,visible=0,screenupdating=0):
pythoncom.CoInitialize()
self.app=gencache.EnsureDispatch(WORD)
self.app.Visible = visible
self.app.DisplayAlerts = 0
self.app.ScreenUpdating = screenupdating
print 'Starting word'
def open(self,doc):
self.opendoc=os.path.basename(doc)
self.app.Documents.Open(FileName=doc)
def replace(self,source,target):
if target=='':target=' '
alltext=self.app.Documents(self.opendoc).Range(Start=0,End=self.app.Documents(self.opendoc).Characters.Count) #select all
alltext.Find.Text = source
alltext.Find.Replacement.Text = target
alltext.Find.Execute(Replace=1,Forward=True)
#Special handling to do replace in text boxes
#http://word.tips.net/Pages/T003879_Updating_a_Field_in_a_Text_Box.html
for shp in self.app.Documents(self.opendoc).Shapes:
if shp.TextFrame.HasText:
shp.TextFrame.TextRange.Find.Text = source
shp.TextFrame.TextRange.Find.Replacement.Text = target
shp.TextFrame.TextRange.Find.Execute(Replace=1,Forward=True)
#My Usage
word=Word(visible=1,screenupdating=1)
word.open(r'C:\Invoice Automation\testTB.doc')
word.replace('[PGN]','1')
Run Code Online (Sandbox Code Playgroud)
self.app .. 部分中的 for shp 是我尝试点击文本框的尝试。它似乎找到了文本框,但它没有替换任何内容。
当我将文本框添加到 Word 文档时,它们被添加到绘图画布中。因此顶级形状是画布,文本框包含在画布中。您应该使用该CanvasItems
方法访问画布中的对象,即文本框
下面的例子对我有用。我创建了一个带有单个文本框的 Word 文档。
import win32com.client
word = win32com.client.Dispatch("Word.Application")
canvas = word.ActiveDocument.Shapes[0]
for item in canvas.CanvasItems:
print item.TextFrame.TextRange.Text
Run Code Online (Sandbox Code Playgroud)
更新:回答 OP 的评论。
我认为您的代码的问题在于每一行代码都Find
创建了一个新Find
对象。您必须创建一个Find
对象并将其绑定到一个名称,然后修改其属性并执行它。所以在你的代码中你应该有:
find = shp.TextFrame.TextRange.Find
find.Text = source
find.Replacement.Text = target
find.Execute(Replace=1, Forward=True)
Run Code Online (Sandbox Code Playgroud)
或单行:
shp.TextFrame.TextRange.Find.Execute(FindText=source, ReplaceWith=target, Replace=1, Forward=True)
Run Code Online (Sandbox Code Playgroud)
这两种方法都适用于我的测试代码。