WebKit 10.8:如何以编程方式设置输入[type = file]字段的值?

时间:2013-01-18 18:15:43

标签: objective-c cocoa webkit osx-mountain-lion html-form

设置

  • Mac OS X 10.8.2
  • 普通Cocoa应用程序链接OS提供的WebKit框架版本。具体而言,/System/Library/Frameworks/WebKit.framework/Versions/A/Resources/version.plist的内容为:
    <plist version="1.0">
    <dict>
        <key>BuildVersion</key>
        <string>5</string>
        <key>CFBundleShortVersionString</key>
        <string>8536</string>
        <key>CFBundleVersion</key>
        <string>8536.26.14</string>
        <key>ProjectName</key>
        <string>WebKit</string>
        <key>SourceVersion</key>
        <string>7536026014000000</string>
    </dict>
    </plist>

我有一个基于WebKit的Cocoa应用程序,它在WebView中加载包含普通HTML表单的HTML文档。 HTML表单包含一个文件选择器输入字段,如下所示:

<form name="foo">
    <input type="file" name="bar">
</form>

我想以编程方式设置此文件选择器的值(如果可能,从Objective-C开始,但我会做任何有效的事情)。

据我所知,这种方法从未起作用:

DOMHTMLInputElement *inputEl = ... // fetch input element
[inputEl setValue:@"some/file.txt"];

我认为WebKit中存在一些安全限制/策略阻止其工作。我认为这是WebKit中的故意安全功能,而不是错误。


然而,在OS X 10.7 Lion中,我 能够通过一些黑客来解决这个限制。

在Lion中,您可以通过编程方式点击 input元素:

#pragma mark -
#pragma mark WebFrameLoadDelegate

- (void)webView:(WebView *)wv didFinishLoadForFrame:(WebFrame *)frame {
    if (frame != [wv mainFrame]) return;

    DOMAbstractView *win = (id)[frame windowObject];
    DOMDocument *doc = [win document];
    DOMHTMLFormElement *formEl = (id)[[doc forms] namedItem:@"foo"];
    DOMHTMLInputElement *inputEl = (id)[[formEl elements] namedItem:@"bar"];
    [inputEl click];
}

哪会产生对-[WebUIDelegate webView:runOpenPanelForFileButtonWithResultListener:]的调用。然后,您可以实现该委托方法以编程方式立即设置结果侦听器的值:

#pragma mark -
#pragma mark WebUIDelegate

- (void)webView:(WebView *)wv runOpenPanelForFileButtonWithResultListener:(id<WebOpenPanelResultListener>)listener {
    [listener chooseFilename:@"some/file.txt"];
}

虽然这是一个丑陋的黑客,但它运作得很好。它具有即时设置文件上载input元素的值的预期效果。屏幕上不会显示“打开面板”。


我担心WebKit已经停止在10.8附带的版本中允许这个。已更改的部分是:

[inputEl click];

这不再对使用10.8的WebKit发货中的元素产生点击事件。

我尝试了其他点击方法,这些方法适用于10.7,但不再适用于10.8:

DOMUIEvent *evt = (id)[doc createEvent:@"UIEvents"];
[evt initUIEvent:@"click" canBubble:YES cancelable:YES view:win detail:1];
[inputEl dispatchEvent:evt];

这些方法在10.7中有效,但在10.8中都没有。


所以,有没有办法在10.8中以编程方式设置此文件选择器的值?

注意:我不要想要随我的应用发送自定义版本的WebKit。除此之外,我对任何建议(ObjC或JS或其他)持开放态度。

如何以编程方式设置10.8附带的WebKit中文件选择器的值?

我有一个示例测试项目(简化测试用例),方便您使用:http://tod.nu/FileUploadTest.zip

1 个答案:

答案 0 :(得分:7)

如果您通过AppKit事件系统调度事件,这将有效。类似的东西:

NSView *docView = [[[webView mainFrame] frameView] documentView];
NSRect docFrame = [docView frame];

NSPoint point = [el boundingBox].origin;
point.y = docFrame.size.height - point.y;

NSEvent *evt = [NSEvent mouseEventWithType:NSLeftMouseDown location:point modifierFlags:0 timestamp:[[NSDate date] timeIntervalSinceReferenceDate] windowNumber:[self.webView.window windowNumber] context:0 eventNumber:0 clickCount:1 pressure:0];
[self.webView.window sendEvent:evt];

evt = [NSEvent mouseEventWithType:NSLeftMouseUp location:point modifierFlags:0 timestamp:[[NSDate date] timeIntervalSinceReferenceDate] windowNumber:[self.webView.window windowNumber] context:0 eventNumber:0 clickCount:1 pressure:0];
[self.webView.window sendEvent:evt];

可能有更好的方法来进行坐标系转换,但这并不是真正重要的部分。