22

给定以下代码:

<input data-bind="event: { change: uploadImage(this.files[0]) }" style="width: 10px;" type="file">

我收到一条错误消息,提示“未定义文件”。我正在尝试转换此演示:

https://github.com/paulrouget/miniuploader/blob/gh-pages/index.html

对淘汰赛友好的实施。当我加载页面时发生错误。如果用户指定了文件,我知道如何访问该文件吗?

jsfiddle:http: //jsfiddle.net/LkqTU/9597/

4

3 回答 3

44

你有两个问题:

  • 如果您只是在对象文字中编写函数调用(uploadImage(this.files[0])),它将在对象文字初始化时执行一次,因此当 KO 解析绑定时。因此它将使用错误的参数执行一次,并且您更改事件将不起作用。您可以通过将其包装到匿名函数中来使其工作。请参阅文档访问事件对象或传递更多参数部分。

  • this不是指您需要使用的绑定中的当前元素$element

所以正确的绑定看起来像这样:

data-bind="event: { change: function() { uploadImage($element.files[0]) } }"

演示JSFiddle

于 2013-06-05T05:06:23.463 回答
20

对于任何感兴趣的人,您可以使用以下自定义绑定,它允许将文件输入元素绑定到包含File的淘汰赛 observable 。

它处理将 observable 设置为所选文件(由 @nemesv 回答),以及在 observable 设置为 null 时清除输入元素(请参阅答案):

ko.bindingHandlers.fileUpload = {
    init: function (element, valueAccessor) {
        $(element).change(function () {
            valueAccessor()(element.files[0]);
        });
    },
    update: function (element, valueAccessor) {
        if (ko.unwrap(valueAccessor()) === null) {
            $(element).wrap('<form>').closest('form').get(0).reset();
            $(element).unwrap();
        }
    }
};

例子:

function Example() {
  var self = this;

  self.uploadFile = ko.observable(null);
  self.uploadName = ko.computed(function() {
    return !!self.uploadFile() ? self.uploadFile().name : '-';
  });

  self.clear = function() {
    self.uploadFile(null);
  };
};

ko.bindingHandlers.fileUpload = {
  init: function(element, valueAccessor) {
    $(element).change(function() {
      valueAccessor()(element.files[0]);
    });
  },
  update: function(element, valueAccessor) {
    if (ko.unwrap(valueAccessor()) === null) {
      $(element).wrap('<form>').closest('form').get(0).reset();
      $(element).unwrap();
    }
  }
};

ko.applyBindings(new Example());
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/knockout/3.2.0/knockout-min.js"></script>

<input type="file" data-bind="fileUpload: uploadFile">
<br/>
<br/>Selected file name: <span data-bind="text: uploadName"></span>
<br/>
<button data-bind="click: clear">Clear input</button>

于 2016-03-04T15:56:14.877 回答
0

大概这个是最简单的了,上传前也可以先预览一下图片

<input type="file" data-bind="fileSrc: src" id="file"/>
<img data-bind="attr:{src:src}"/>

$(function() {
    ko.applyBindings(new ViewModel());
});

var ViewModel = function () {
    var self = this;

    self.src = ko.observable();
};

ko.bindingHandlers.fileSrc = {
    init: function (element, valueAccessor) {
        ko.utils.registerEventHandler(element, "change", function () {
            var reader = new FileReader();

            reader.onload = function (e) {
                var value = valueAccessor();
                value(e.target.result);
            }

            reader.readAsDataURL(element.files[0]);
        });
    }
};
于 2021-01-30T00:15:07.983 回答