AngularJs: Как проверить изменения в полях ввода файлов?

240

Я новичок в angular. Я пытаюсь прочитать загруженный путь к файлу из поля "файл" HTML всякий раз, когда в этом поле происходит "изменение". Если я использую 'onChange', он работает, но когда я использую его angular, используя 'ng-change', он не работает.

<script>
   var DemoModule = angular.module("Demo",[]);
   DemoModule .controller("form-cntlr",function($scope){
   $scope.selectFile = function()
   {
        $("#file").click();
   }
   $scope.fileNameChaged = function()
   {
        alert("select file");
   }
});
</script>

<div ng-controller="form-cntlr">
    <form>
         <button ng-click="selectFile()">Upload Your File</button>
         <input type="file" style="display:none" 
                          id="file" name='file' ng-Change="fileNameChaged()"/>
    </form>  
</div>

fileNameChaged() никогда не вызывает. Firebug также не показывает никаких ошибок.

Теги:

14 ответов

201
Лучший ответ

Отсутствует поддержка привязки для управления загрузкой файлов

https://github.com/angular/angular.js/issues/1375

<div ng-controller="form-cntlr">
        <form>
             <button ng-click="selectFile()">Upload Your File</button>
             <input type="file" style="display:none" 
                id="file" name='file' onchange="angular.element(this).scope().fileNameChanged(this)" />
        </form>  
    </div>

вместо

 <input type="file" style="display:none" 
    id="file" name='file' ng-Change="fileNameChanged()" />

можете попробовать

<input type="file" style="display:none" 
    id="file" name='file' onchange="angular.element(this).scope().fileNameChanged()" />

Примечание. Для этого требуется, чтобы приложение angular всегда находилось в режиме отладки. Это не будет работать в производственном коде, если режим отладки отключен.

и в вашей функции изменения вместо

$scope.fileNameChanged = function() {
   alert("select file");
}

можете попробовать

$scope.fileNameChanged = function() {
  console.log("select file");
}

Ниже приведен один рабочий пример загрузки файла с загрузкой файла перетаскивания. Может оказаться полезным http://jsfiddle.net/danielzen/utp7j/

Angular Информация о загрузке файлов

URL для загрузки файла AngularJS в ASP.Net

http://cgeers.com/2013/05/03/angularjs-file-upload/

Загрузка нескольких файлов с несколькими файлами AngularJs с помощью NodeJS

http://jasonturim.wordpress.com/2013/09/12/angularjs-native-multi-file-upload-with-progress/

ngUpload - служба AngularJS для загрузки файлов с использованием iframe

http://ngmodules.org/modules/ngUpload

  • 0
    Это сработало!!! Еще один вопрос, как я могу получить путь к файлу из этого типа ввода файла?
  • 2
    onchange = "angular.element (this) .scope (). fileNameChaged (this)" $ scope.fileNameChaged = function (element) {console.log ('files:', element.files); } Можете ли вы поменять в двух местах и проверить? в зависимости от браузера вы получите полный путь к файлу. У меня есть обновленная скрипта ниже, это файл загрузки URL и проверка консоли jsfiddle.net/utp7j/260
Показать ещё 16 комментариев
389

Я сделал небольшую директиву для прослушивания изменений ввода файла.

Просмотр JSFiddle

view.html:

<input type="file" custom-on-change="uploadFile">

controller.js:

app.controller('myCtrl', function($scope){
    $scope.uploadFile = function(event){
        var files = event.target.files;
    };
});     

directive.js:

app.directive('customOnChange', function() {
  return {
    restrict: 'A',
    link: function (scope, element, attrs) {
      var onChangeHandler = scope.$eval(attrs.customOnChange);
      element.on('change', onChangeHandler);
      element.on('$destroy', function() {
        element.off();
      });

    }
  };
});
  • 1
    Это хорошо работает, если каждый раз выбирается другой файл. Можно ли слушать, когда выбран один и тот же файл?
  • 4
    @JDawg Вам нужно сбросить входное значение jsfiddle.net/sqren/7t0h5aax
Показать ещё 21 комментарий
31

Это уточнение некоторых из других, данные будут в конечном итоге в ng-модели, которая обычно вам нужна.

Разметка (просто создайте файл данных атрибута, чтобы директива могла его найти)

<input
    data-file
    id="id_image" name="image"
    ng-model="my_image_model" type="file">

JS

app.directive('file', function() {
    return {
        require:"ngModel",
        restrict: 'A',
        link: function($scope, el, attrs, ngModel){
            el.bind('change', function(event){
                var files = event.target.files;
                var file = files[0];

                ngModel.$setViewValue(file);
                $scope.$apply();
            });
        }
    };
});
  • 1
    Является ли $scope.$apply(); требуется? Кажется, что мое событие ng-change все еще происходит без него.
  • 0
    @ row1, вам нужно только вручную запустить заявку, если у вас есть другие угловатые вещи, которые нужно знать об этом во время этой операции.
26

Чистым способом является создание собственной директивы для привязки к событию "change". Просто чтобы вы знали, что IE9 не поддерживает FormData, поэтому вы не можете реально получить объект файл из события изменения.

Вы можете использовать ng-file-upload библиотеку, которая уже поддерживает IE с FileAPI polyfill и упрощает отправку файла на сервер. Он использует директиву для достижения этого.

<script src="angular.min.js"></script>
<script src="ng-file-upload.js"></script>

<div ng-controller="MyCtrl">
  <input type="file" ngf-select="onFileSelect($files)" multiple>
</div>

JS:

//inject angular file upload directive.
angular.module('myApp', ['ngFileUpload']);

var MyCtrl = [ '$scope', 'Upload', function($scope, Upload) {
  $scope.onFileSelect = function($files) {
    //$files: an array of files selected, each file has name, size, and type.
    for (var i = 0; i < $files.length; i++) {
      var $file = $files[i];
      Upload.upload({
        url: 'my/upload/url',
        data: {file: $file}
      }).then(function(data, status, headers, config) {
        // file is uploaded successfully
        console.log(data);
      }); 
    }
  }
}];
  • 0
    Этот пример устарел. Получить новый из документов здесь .
22

Я расширил идею @Stuart Axon, чтобы добавить двустороннюю привязку для ввода файла (т.е. разрешить сброс ввода путем сброса значения модели обратно до нуля):

app.directive('bindFile', [function () {
    return {
        require: "ngModel",
        restrict: 'A',
        link: function ($scope, el, attrs, ngModel) {
            el.bind('change', function (event) {
                ngModel.$setViewValue(event.target.files[0]);
                $scope.$apply();
            });

            $scope.$watch(function () {
                return ngModel.$viewValue;
            }, function (value) {
                if (!value) {
                    el.val("");
                }
            });
        }
    };
}]);

Демо

  • 0
    Спасибо @JLRishe. Одно замечание, на самом деле data-ng-click="vm.theFile=''" работает хорошо, нет необходимости записывать его в метод контроллера
16

Как и некоторые другие хорошие ответы здесь, я написал директиву для решения этой проблемы, но эта реализация более точно отражает способ angular привязки событий.

Вы можете использовать следующую директиву:

HTML

<input type="file" file-change="yourHandler($event, files)" />

Как вы можете видеть, вы можете вставлять файлы, выбранные в обработчик событий, так как вы добавляете объект $event в любой обработчик событий ng.

Javascript

angular
  .module('yourModule')
  .directive('fileChange', ['$parse', function($parse) {

    return {
      require: 'ngModel',
      restrict: 'A',
      link: function ($scope, element, attrs, ngModel) {

        // Get the function provided in the file-change attribute.
        // Note the attribute has become an angular expression,
        // which is what we are parsing. The provided handler is 
        // wrapped up in an outer function (attrHandler) - we'll 
        // call the provided event handler inside the handler()
        // function below.
        var attrHandler = $parse(attrs['fileChange']);

        // This is a wrapper handler which will be attached to the
        // HTML change event.
        var handler = function (e) {

          $scope.$apply(function () {

            // Execute the provided handler in the directive scope.
            // The files variable will be available for consumption
            // by the event handler.
            attrHandler($scope, { $event: e, files: e.target.files });
          });
        };

        // Attach the handler to the HTML change event 
        element[0].addEventListener('change', handler, false);
      }
    };
  }]);
  • 0
    боролся с этим пару дней, пока не попробовал ваш ответ. Спасибо!
  • 0
    Как я могу передать дополнительный параметр в выражение fileChange ? Я использую эту директиву внутри ng-repeat и переменная $scope передаваемая в attrHandler одинакова для каждой записи. Какое лучшее решение?
12

Эта директива также передает выбранные файлы:

/**
 *File Input - custom call when the file has changed
 */
.directive('onFileChange', function() {
  return {
    restrict: 'A',
    link: function (scope, element, attrs) {
      var onChangeHandler = scope.$eval(attrs.onFileChange);

      element.bind('change', function() {
        scope.$apply(function() {
          var files = element[0].files;
          if (files) {
            onChangeHandler(files);
          }
        });
      });

    }
  };
});

HTML, как его использовать:

<input type="file" ng-model="file" on-file-change="onFilesSelected">

В моем контроллере:

$scope.onFilesSelected = function(files) {
     console.log("files - " + files);
};
  • 1
    Я получаю onChangeHandler не ошибка функции?
  • 0
    Как выглядит $ scope.onFilesSelected в вашем контроллере?
Показать ещё 4 комментария
3

Я рекомендую создать директиву

<input type="file" custom-on-change handler="functionToBeCalled(params)">

app.directive('customOnChange', [function() {
        'use strict';

        return {
            restrict: "A",

            scope: {
                handler: '&'
            },
            link: function(scope, element){

                element.change(function(event){
                    scope.$apply(function(){
                        var params = {event: event, el: element};
                        scope.handler({params: params});
                    });
                });
            }

        };
    }]);

эта директива может использоваться много раз, она использует свою собственную область действия и не зависит от родительской области. Вы также можете дать некоторые параметры функции обработчика. Функция обработчика будет вызвана с объектом scope, который был активен при изменении ввода. $ применяются обновления вашей модели каждый раз, когда событие изменения называется

2

Самая простая версия Angular jqLite.

JS:

.directive('cOnChange', function() {
    'use strict';

    return {
        restrict: "A",
        scope : {
            cOnChange: '&'
        },
        link: function (scope, element) {
            element.on('change', function () {
                scope.cOnChange();
        });
        }
    };
});

HTML:

<input type="file" data-c-on-change="your.functionName()">
  • 0
    Если файл открыт и когда мы пытаемся загрузить файл в браузере Microsoft Edge. Ничего не происходит. Вы можете помочь?
0
Элементы

Angular (например, корневой элемент директивы) являются объектами jQuery [Lite]. Это означает, что мы можем зарегистрировать прослушиватель событий так:

link($scope, $el) {
    const fileInputSelector = '.my-file-input'

    function setFile() {
        // access file via $el.find(fileInputSelector).get(0).files[0]
    }

    $el.on('change', fileInputSelector, setFile)
}

Это событие события jQuery. Здесь слушатель присоединяется к корневому элементу директивы. Когда событие будет запущено, оно будет пузыриться до зарегистрированного элемента, и jQuery определит, возникло ли событие от внутреннего элемента, соответствующего определенному селектору. Если это произойдет, обработчик будет срабатывать.

Преимущества этого метода:

  • обработчик привязан к элементу $, который будет автоматически очищен при уничтожении области действия.
  • в шаблоне нет кода
  • будет работать, даже если целевой делегат (ввод) еще не был отображен при регистрации обработчика события (например, при использовании ng-if или ng-switch)

http://api.jquery.com/on/

0

Рабочая демонстрация директивы "файлы-ввод", которая работает с ng-change 1

Чтобы создать элемент <input type=file>, выполните директиву ng-change, ему потребуется настраиваемая директива, которая работает с директивой ng-model.

<input type="file" files-input ng-model="fileList" 
       ng-change="onInputChange()" multiple />

DEMO

angular.module("app",[])
.directive("filesInput", function() {
  return {
    require: "ngModel",
    link: function postLink(scope,elem,attrs,ngModel) {
      elem.on("change", function(e) {
        var files = elem[0].files;
        ngModel.$setViewValue(files);
      })
    }
  }
})

.controller("ctrl", function($scope) {
     $scope.onInputChange = function() {
         console.log("input change");
     };
})
<script src="//unpkg.com/angular/angular.js"></script>
  <body ng-app="app" ng-controller="ctrl">
    <h1>AngularJS Input `type=file` Demo</h1>
    
    <input type="file" files-input ng-model="fileList" 
           ng-change="onInputChange()" multiple />
    
    <h2>Files</h2>
    <div ng-repeat="file in fileList">
      {{file.name}}
    </div>
  </body>
0

Другим интересным способом прослушивания изменений ввода файла является просмотр атрибута ng-model входного файла. Off course FileModel - это настраиваемая директива.

Вот так:

HTML → <input type="file" file-model="change.fnEvidence">

Код JS →

$scope.$watch('change.fnEvidence', function() {
                    alert("has changed");
                });

Надеюсь, это поможет кому-то.

0

Я сделал это вот так:

<!-- HTML -->
<button id="uploadFileButton" class="btn btn-info" ng-click="vm.upload()">    
<span  class="fa fa-paperclip"></span></button>
<input type="file" id="txtUploadFile" name="fileInput" style="display: none;" />
// self is the instance of $scope or this
self.upload = function () {
   var ctrl = angular.element("#txtUploadFile");
   ctrl.on('change', fileNameChanged);
   ctrl.click();
}

function fileNameChanged(e) {
    console.log(self.currentItem);
    alert("select file");
}
0

Слишком полная база решений:

`onchange="angular.element(this).scope().UpLoadFile(this.files)"`

Простой способ скрыть поле ввода и заменить его изображением, здесь после решения, которое также требует взломать angular, но выполняющего задание [TriggerEvent не работает должным образом]

Решение:

  • поместите поле ввода на дисплей: none [поле ввода существует в DOM, но не отображается]
  • разместите свое изображение сразу после На изображении используйте nb-click() для активации метода

При щелчке изображения имитируется действие DOM 'click' в поле ввода. Et voilà!

 var tmpl = '<input type="file" id="{{name}}-filein"' + 
             'onchange="angular.element(this).scope().UpLoadFile(this.files)"' +
             ' multiple accept="{{mime}}/*" style="display:none" placeholder="{{placeholder}}">'+
             ' <img id="{{name}}-img" src="{{icon}}" ng-click="clicked()">' +
             '';
   // Image was clicked let simulate an input (file) click
   scope.inputElem = elem.find('input'); // find input in directive
   scope.clicked = function () {
         console.log ('Image clicked');
         scope.inputElem[0].click(); // Warning Angular TriggerEvent does not work!!!
    };
  • 0
    Вы не можете использовать angular.element (this) .scope (), это функция отладки!

Ещё вопросы

Сообщество Overcoder
Наверх
Меню