我试图使用QtWebKit模块上传照片到vk.com。我面临的问题是无法正确填充input(type="file")
的值。下面是我使用的一些相关代码:
def upload():
print 'uploading...'
photoInput = web.page().mainFrame().documentElement().findFirst('input[id="photos_upload_input"]')
assert photoInput, 'No input found'
photoInput.setAttribute('value', '/Users/elmigranto/Downloads/stuff.png')
print photoInput.evaluateJavaScript('return this.value;').toString()
值得注意的是,由于浏览器的安全策略,Javascript不可能填充文件输入的值。然而,应该可以使用Qt API,更具体地说,QWebElement
::
setAttribute()
方法。这就是我所做的……没有效果(好吧,photoInput.attribute('value')
返回预期结果,但photoInput.evaluateJavaScript('return this.value;').toString()
返回空字符串,输入的onchange
处理程序也没有触发)。
设置其他属性没有问题,例如,QWebElement
::
addClass()
就像一个魅力。
谢谢。
由于安全原因,setAttribute
方法可能仍然无法工作。
但是您可以重新定义函数QWebPage::chooseFile
,它通常应该打开上传对话框并返回文件名,以便它在不打开对话框的情况下返回静态文件名,并通过模拟输入元素上的"返回"键来激活上传。
这似乎可以工作:
from PyQt4.QtCore import *
from PyQt4.QtGui import *
from PyQt4.QtWebKit import *
import sys
class WebPage(QWebPage):
def __init__(self, parent = None):
super(WebPage, self).__init__(parent)
self.overrideUpload = None
def chooseFile(self, originatingFrame, oldFile):
if self.overrideUpload is None:
return super(WebPage, self).chooseFile(originatingFrame, oldFile)
result = self.overrideUpload
self.overrideUpload = None
return result
def setUploadFile(self, selector, filename):
button = self.mainFrame().documentElement().findFirst(selector)
self.overrideUpload = filename
# set the focus on the input element
button.setFocus();
# and simulate a keypress event to make it call our chooseFile method
webview.event(QKeyEvent(QEvent.KeyPress, Qt.Key_Enter, Qt.NoModifier))
def upload():
print 'uploading...'
page.setUploadFile('input[id="photos_upload_input"]',
'/Users/elmigranto/Downloads/stuff.png')
# The change seems to be asynchronous, at it isn't visible
# just after the previous call
app = QApplication(sys.argv)
webview = QWebView()
page = WebPage(webview)
webview.setPage(page)
source = '''
<form action="#">
Select a file: <input type="file" id="photos_upload_input">
<input type="submit">
</form>
'''
webview.loadFinished.connect(upload)
webview.show()
webview.setHtml(source)
sys.exit(app.exec_())