Html-помічник для <input type = "file" />


124

Чи HTMLHelperможна завантажити файл? Зокрема, я шукаю заміну

<input type="file"/>

за допомогою ASP.NET MVC HTMLHelper.

Або, якщо я використовую

using (Html.BeginForm()) 

Що таке контроль HTML для завантаження файлу?

Відповіді:


207

HTML файл завантаження ASP MVC 3.

Модель : ( Зверніть увагу, що FileExtensionsAttribute доступний у MvcFutures. Це підтвердить розширення файлів на стороні клієнта та стороні сервера. )

public class ViewModel
{
    [Required, Microsoft.Web.Mvc.FileExtensions(Extensions = "csv", 
             ErrorMessage = "Specify a CSV file. (Comma-separated values)")]
    public HttpPostedFileBase File { get; set; }
}

Перегляд HTML :

@using (Html.BeginForm("Action", "Controller", FormMethod.Post, new 
                                       { enctype = "multipart/form-data" }))
{
    @Html.TextBoxFor(m => m.File, new { type = "file" })
    @Html.ValidationMessageFor(m => m.File)
}

Дія контролера :

[HttpPost]
public ActionResult Action(ViewModel model)
{
    if (ModelState.IsValid)
    {
        // Use your file here
        using (MemoryStream memoryStream = new MemoryStream())
        {
            model.File.InputStream.CopyTo(memoryStream);
        }
    }
}

Це не відображає введення файлу <input type="file" />, а лише текстове поле
Бен

@PauliusZaliaduonis у рядку Microsoft.Web.Mvc.FileExtensions MVC підкреслено червоним кольором. Як я можу це виправити?
Pomster

1
@pommy Зауважте, що FileExtensionsAttribute доступний у MvcFutures (Станом на MVC3). Ви можете використовувати джерела звідси: Джерела чи це доступно в .NET Framework 4.5, див. Документацію MSDN
Paulius Zaliaduonis

1
На жаль, атрибут FileExtension, здається, не працює з властивостями типу HttpPostedFileBase, а скоріше здається лише рядковим. Принаймні, він ніколи не приймав pdf як дійсне розширення.
Serj Sagan

Це додасть атрибут значення (value = ""), який не підтверджується як дійсний HTML5. значення не є дійсним для файлів та зображень типів введення. Я не бачу жодного способу видалення атрибута значення. Здається, це жорстко закодовано.
Dan Friedman

19

Ви також можете використовувати:

@using (Html.BeginForm("Upload", "File", FormMethod.Post, new { enctype = "multipart/form-data" }))
{ 
    <p>
        <input type="file" id="fileUpload" name="fileUpload" size="23" />
    </p>
    <p>
        <input type="submit" value="Upload file" /></p> 
}

8

У мене це питання було деякий час назад, і я натрапив на один із дописів Скотта Хензельмана:

Реалізація завантаження файлів HTTP за допомогою ASP.NET MVC, включаючи тести та макети

Сподіваюся, це допомагає.


Дякую, але я спеціально шукаю реалізацію використання (Html.BeginForm ()), а не інших варіантів.
Гравітон

6

Або ви могли зробити це правильно:

У вашому класі HtmlHelper Extension:

public static MvcHtmlString FileFor<TModel, TProperty>(this HtmlHelper<TModel> helper, Expression<Func<TModel, TProperty>> expression)
    {
        return helper.FileFor(expression, null);
    }

public static MvcHtmlString FileFor<TModel, TProperty>(this HtmlHelper<TModel> helper, Expression<Func<TModel, TProperty>> expression, object htmlAttributes)
    {
        var builder = new TagBuilder("input");

        var id = helper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(ExpressionHelper.GetExpressionText(expression));
        builder.GenerateId(id);
        builder.MergeAttribute("name", id);
        builder.MergeAttribute("type", "file");

        builder.MergeAttributes(new RouteValueDictionary(htmlAttributes));

        // Render tag
        return MvcHtmlString.Create(builder.ToString(TagRenderMode.SelfClosing));
    }

Цей рядок:

var id = helper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(ExpressionHelper.GetExpressionText(expression));

Створює унікальний для моделі ідентифікатор, який ви знаєте у списках та ін. модель [0]. Ім'я тощо.

Створіть правильну властивість у моделі:

public HttpPostedFileBase NewFile { get; set; }

Тоді вам потрібно переконатися, що ваша форма надсилатиме файли:

@using (Html.BeginForm("Action", "Controller", FormMethod.Post, new { enctype = "multipart/form-data" }))

Тоді ось ваш помічник:

@Html.FileFor(x => x.NewFile)

Це рішення є більш цукеркою для очей, і підтримує мене у відповідності з допоміжними методами @Html.
Яхфуфі

4

Удосконалена версія відповіді Поліуса Заліадуніса:

Для правильної роботи валідації довелося змінити Модель на:

public class ViewModel
{
      public HttpPostedFileBase File { get; set; }

        [Required(ErrorMessage="A header image is required"), FileExtensions(ErrorMessage = "Please upload an image file.")]
        public string FileName
        {
            get
            {
                if (File != null)
                    return File.FileName;
                else
                    return String.Empty;
            }
        }
}

і погляд на:

@using (Html.BeginForm("Action", "Controller", FormMethod.Post, new 
                                       { enctype = "multipart/form-data" }))
{
    @Html.TextBoxFor(m => m.File, new { type = "file" })
    @Html.ValidationMessageFor(m => m.FileName)
}

Це потрібно тому, що те, що @Serj Sagan писав про атрибут FileExtension, працюючи лише з рядками.


Ви не можете злити цю відповідь у відповідь Павлія?
Гравітон

2

Щоб скористатися BeginForm, ось спосіб його використання:

 using(Html.BeginForm("uploadfiles", 
"home", FormMethod.POST, new Dictionary<string, object>(){{"type", "file"}})

2
Спочатку ви згадуєте, як генерувати елемент введення, а тепер ви говорите про те, як генерувати елемент форми? Це справді ваша відповідь?
pupeno

0

Це також працює:

Модель:

public class ViewModel
{         
    public HttpPostedFileBase File{ get; set; }
}

Вид:

@using (Html.BeginForm("Action", "Controller", FormMethod.Post, new 
                                       { enctype = "multipart/form-data" }))
{
    @Html.TextBoxFor(m => m.File, new { type = "file" })       
}

Дія контролера:

[HttpPost]
public ActionResult Action(ViewModel model)
{
    if (ModelState.IsValid)
    {
        var postedFile = Request.Files["File"];

       // now you can get and validate the file type:
        var isFileSupported= IsFileSupported(postedFile);

    }
}

public bool IsFileSupported(HttpPostedFileBase file)
            {
                var isSupported = false;

                switch (file.ContentType)
                {

                    case ("image/gif"):
                        isSupported = true;
                        break;

                    case ("image/jpeg"):
                        isSupported = true;
                        break;

                    case ("image/png"):
                        isSupported = true;
                        break;


                    case ("audio/mp3"):  
                        isSupported = true;
                        break;

                    case ("audio/wav"):  
                        isSupported = true;
                        break;                                 
                }

                return isSupported;
            }

Список типів вмісту


-2

Думаю, це трохи хакі, але це призводить до застосування правильних атрибутів перевірки тощо

@Html.Raw(Html.TextBoxFor(m => m.File).ToHtmlString().Replace("type=\"text\"", "type=\"file\""))
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.