Passando uma lista de objetos para um método de controlador MVC usando jQuery Ajax

113

Estou tentando passar uma matriz de objetos em um método de controlador MVC usando a função ajax () do jQuery. Quando entro no método do controlador PassThing () C #, o argumento "coisas" é nulo. Eu tentei isso usando um tipo de lista para o argumento, mas também não funciona. O que estou fazendo de errado?

<script type="text/javascript">
    $(document).ready(function () {
        var things = [
            { id: 1, color: 'yellow' },
            { id: 2, color: 'blue' },
            { id: 3, color: 'red' }
        ];

        $.ajax({
            contentType: 'application/json; charset=utf-8',
            dataType: 'json',
            type: 'POST',
            url: '/Xhr/ThingController/PassThing',
            data: JSON.stringify(things)
        });
    });
</script>

public class ThingController : Controller
{
    public void PassThing(Thing[] things)
    {
        // do stuff with things here...
    }

    public class Thing
    {
        public int id { get; set; }
        public string color { get; set; }
    }
}
idílico
fonte
3
Seus dados são uma string, mas seu método aceita uma matriz. Altere seu método para aceitar uma string e, em seguida, desserialize-o dentro do método.
Bob Horn
2
Seu código está correto. Eu testei e funcionou usando MVC 4. Forneça mais dados para descobrir.
Diego
Isso é ótimo, mas e se você não precisar apenas de uma lista de strings para passar, mas precisar incluir um id separado associado à lista de strings? Por exemplo, id do grupo, lista de grupos sob o id do grupo.
Nathan McKaskle de

Respostas:

188

Usando a sugestão de NickW, consegui fazer isso funcionar usando things = JSON.stringify({ 'things': things });Aqui está o código completo.

$(document).ready(function () {
    var things = [
        { id: 1, color: 'yellow' },
        { id: 2, color: 'blue' },
        { id: 3, color: 'red' }
    ];      

    things = JSON.stringify({ 'things': things });

    $.ajax({
        contentType: 'application/json; charset=utf-8',
        dataType: 'json',
        type: 'POST',
        url: '/Home/PassThings',
        data: things,
        success: function () {          
            $('#result').html('"PassThings()" successfully called.');
        },
        failure: function (response) {          
            $('#result').html(response);
        }
    }); 
});


public void PassThings(List<Thing> things)
{
    var t = things;
}

public class Thing
{
    public int Id { get; set; }
    public string Color { get; set; }
}

Aprendi duas coisas com isso:

  1. As configurações contentType e dataType são absolutamente necessárias na função ajax (). Não funcionará se eles estiverem faltando. Descobri isso depois de muita tentativa e erro.

  2. Para passar uma matriz de objetos para um método de controlador MVC, simplesmente use o formato JSON.stringify ({'coisas': coisas}).

Espero que isto ajude alguém!

idílico
fonte
8
Eu estava tendo o mesmo problema e adicionar o contentType corrigiu isso. Obrigado!
Rochelle C
9
Duas coisas a serem observadas: JSON.stringify e especificando 'contentType'.
dinesh ygv
Crud. Ainda não está funcionando para mim. meu URL de solicitação é http://localhost:52459/Sales/completeSale?itemsInCart=[{"ItemId":1,"Quantity":"1","Price":3.5}]e Sales.completeSaleé public ActionResult completeSale(ItemInCart[] itemsInCart)anotado como um HttpGet.
abalter
3
por alguma razão eu tive que usardata: JSON.stringify(things),
Rob Scott
1
dataTypenão é necessário. Se for omitido, a função ajax trabalhará com base nos dados de retorno
32

Você não poderia simplesmente fazer isso?

var things = [
    { id: 1, color: 'yellow' },
    { id: 2, color: 'blue' },
    { id: 3, color: 'red' }
];
$.post('@Url.Action("PassThings")', { things: things },
   function () {
        $('#result').html('"PassThings()" successfully called.');
   });

... e marque sua ação com

[HttpPost]
public void PassThings(IEnumerable<Thing> things)
{
    // do stuff with things here...
}
Pântano da Lanterna
fonte
3
Esta deve ser a melhor resposta. O JSON.stringify não deve ser usado neste caso
Isso não está funcionando para mim..Estou usando [HttpPost] public int SaveResults (List <ShortDetail> model) {} e $ .post ("@ Url.Action (" SaveResults "," Maps ")", {model: dataItems}, função (resultado) {});
Samra
2
Funcionou para mim Absolutamente a melhor resposta. Não sei por que a implementação do Halcyon não funcionou. A função PassThings foi chamada, mas a variável de entrada 'coisas' estava vazia, mesmo se tiver sido preenchida no javascript pouco antes da chamada.
Leonardo Daga
12

Estou usando um aplicativo da Web .Net Core 2.1 e não consegui obter uma única resposta aqui para funcionar. Ou obtive um parâmetro em branco (se o método foi chamado) ou um erro 500 do servidor. Comecei a brincar com todas as combinações possíveis de respostas e finalmente consegui um resultado de trabalho.

No meu caso, a solução foi a seguinte:

Script - restringe a matriz original (sem usar uma propriedade nomeada)

    $.ajax({
        type: 'POST',
        contentType: 'application/json; charset=utf-8',
        url: mycontrolleraction,
        data: JSON.stringify(things)
    });

E no método do controlador, use [FromBody]

    [HttpPost]
    public IActionResult NewBranch([FromBody]IEnumerable<Thing> things)
    {
        return Ok();
    }

As falhas incluem:

  • Nomeando o conteúdo

    dados: {conteúdo: nós}, // Erro de servidor 500

  • Não tendo contentType = Erro de servidor 500

Notas

  • dataTypenão é necessário, apesar do que algumas respostas dizem, pois isso é usado para a decodificação da resposta (portanto, não é relevante para os exemplos de solicitação aqui).
  • List<Thing> também funciona no método do controlador
Gone Coding
fonte
10

Eu tenho uma resposta perfeita para tudo isso: eu tentei tantas soluções, mas não consegui me tornar finalmente capaz de gerenciar, por favor encontre a resposta detalhada abaixo:

       $.ajax({
            traditional: true,
            url: "/Conroller/MethodTest",
            type: "POST",
            contentType: "application/json; charset=utf-8",
            data:JSON.stringify( 
               [
                { id: 1, color: 'yellow' },
                { id: 2, color: 'blue' },
                { id: 3, color: 'red' }
                ]),
            success: function (data) {
                $scope.DisplayError(data.requestStatus);
            }
        });

Controler

public class Thing
{
    public int id { get; set; }
    public string color { get; set; }
}

public JsonResult MethodTest(IEnumerable<Thing> datav)
    {
   //now  datav is having all your values
  }
Veera Induvasi
fonte
Você deve ter mais votos positivos: tradicional: verdadeiro é a forma recomendada no site Jquery
DFTR
7

A única maneira de fazer isso funcionar é passar o JSON como uma string e, em seguida, desserializar usando JavaScriptSerializer.Deserialize<T>(string input), o que é muito estranho se esse for o desserializador padrão para MVC 4.

Meu modelo tem listas aninhadas de objetos e o melhor que consegui usando dados JSON é a lista superior para ter o número correto de itens, mas todos os campos nos itens eram nulos.

Esse tipo de coisa não deve ser tão difícil.

    $.ajax({
        type: 'POST',
        url: '/Agri/Map/SaveSelfValuation',
        data: { json: JSON.stringify(model) },
        dataType: 'text',
        success: function (data) {

    [HttpPost]
    public JsonResult DoSomething(string json)
    {
        var model = new JavaScriptSerializer().Deserialize<Valuation>(json);
brincadeira mais urgente
fonte
Para fazer isso funcionar, siga de perto o formato da chamada Ajax.
Graham Laight
4

Este é um código de trabalho para sua consulta, você pode usá-lo.

Controler

    [HttpPost]
    public ActionResult save(List<ListName> listObject)
    {
    //operation return
    Json(new { istObject }, JsonRequestBehavior.AllowGet); }
    }

javascript

  $("#btnSubmit").click(function () {
    var myColumnDefs = [];
    $('input[type=checkbox]').each(function () {
        if (this.checked) {
            myColumnDefs.push({ 'Status': true, 'ID': $(this).data('id') })
        } else {
            myColumnDefs.push({ 'Status': false, 'ID': $(this).data('id') })
        }
    });
   var data1 = { 'listObject': myColumnDefs};
   var data = JSON.stringify(data1)
   $.ajax({
   type: 'post',
   url: '/Controller/action',
   data:data ,
   contentType: 'application/json; charset=utf-8',
   success: function (response) {
    //do your actions
   },
   error: function (response) {
    alert("error occured");
   }
   });
sach4all
fonte
2

Envolver sua lista de objetos com outro objeto contendo uma propriedade que corresponda ao nome do parâmetro que é esperado pelo controlador MVC funciona. A parte importante é o wrapper em torno da lista de objetos.

$(document).ready(function () {
    var employeeList = [
        { id: 1, name: 'Bob' },
        { id: 2, name: 'John' },
        { id: 3, name: 'Tom' }
    ];      

    var Employees = {
      EmployeeList: employeeList
    }

    $.ajax({
        dataType: 'json',
        type: 'POST',
        url: '/Employees/Process',
        data: Employees,
        success: function () {          
            $('#InfoPanel').html('It worked!');
        },
        failure: function (response) {          
            $('#InfoPanel').html(response);
        }
    }); 
});


public void Process(List<Employee> EmployeeList)
{
    var emps = EmployeeList;
}

public class Employee
{
    public int Id { get; set; }
    public string Name { get; set; }
}
Hoodlum
fonte
1
     var List = @Html.Raw(Json.Encode(Model));
$.ajax({
    type: 'post',
    url: '/Controller/action',
    data:JSON.stringify({ 'item': List}),
    contentType: 'application/json; charset=utf-8',
    success: function (response) {
        //do your actions
    },
    error: function (response) {
        alert("error occured");
    }
});
Athul Nalupurakkal
fonte
Tente este código para passar objetos de modelo de lista de usando ajax. O modelo representa o IList <Model>. Use IList <Model> no controlador para obter os valores.
Athul Nalupurakkal
0

Se você estiver usando a API Web ASP.NET, então você deve apenas passar data: JSON.stringify(things).

E seu controlador deve ser parecido com isto:

public class PassThingsController : ApiController
{
    public HttpResponseMessage Post(List<Thing> things)
    {
        // code
    }
}
FleGMan
fonte
0

Modificação de @veeresh i

 var data=[

                        { id: 1, color: 'yellow' },
                        { id: 2, color: 'blue' },
                        { id: 3, color: 'red' }
                        ]; //parameter
        var para={};
        para.datav=data;   //datav from View


        $.ajax({
                    traditional: true,
                    url: "/Conroller/MethodTest",
                    type: "POST",
                    contentType: "application/json; charset=utf-8",
                    data:para,
                    success: function (data) {
                        $scope.DisplayError(data.requestStatus);
                    }
                });

In MVC



public class Thing
    {
        public int id { get; set; }
        public string color { get; set; }
    }

    public JsonResult MethodTest(IEnumerable<Thing> datav)
        {
       //now  datav is having all your values
      }
Minhajul Islam
fonte
0

O que eu fiz ao tentar enviar alguns dados de várias linhas selecionadas em DataTable para ação MVC:

HTML no início de uma página:

@Html.AntiForgeryToken()

(apenas uma linha é mostrada, vincular do modelo):

 @foreach (var item in Model.ListOrderLines)
                {
                    <tr data-orderid="@item.OrderId" data-orderlineid="@item.OrderLineId" data-iscustom="@item.IsCustom">
                        <td>@item.OrderId</td>
                        <td>@item.OrderDate</td>
                        <td>@item.RequestedDeliveryDate</td>
                        <td>@item.ProductName</td>
                        <td>@item.Ident</td>
                        <td>@item.CompanyName</td>
                        <td>@item.DepartmentName</td>
                        <td>@item.ProdAlias</td>
                        <td>@item.ProducerName</td>
                        <td>@item.ProductionInfo</td>
                    </tr>
                }

Botão que inicia a função JavaScript:

 <button class="btn waves-effect waves-light btn-success" onclick="ProcessMultipleRows();">Start</button>

Função JavaScript:

  function ProcessMultipleRows() {
            if ($(".dataTables_scrollBody>tr.selected").length > 0) {
                var list = [];
                $(".dataTables_scrollBody>tr.selected").each(function (e) {
                    var element = $(this);
                    var orderid = element.data("orderid");
                    var iscustom = element.data("iscustom");
                    var orderlineid = element.data("orderlineid");
                    var folderPath = "";
                    var fileName = "";

                    list.push({ orderId: orderid, isCustomOrderLine: iscustom, orderLineId: orderlineid, folderPath: folderPath, fileName : fileName});
                });

                $.ajax({
                    url: '@Url.Action("StartWorkflow","OrderLines")',
                    type: "post", //<------------- this is important
                    data: { model: list }, //<------------- this is important
                    beforeSend: function (xhr) {//<--- This is important
                      xhr.setRequestHeader("RequestVerificationToken",
                      $('input:hidden[name="__RequestVerificationToken"]').val());
                      showPreloader();
                    },
                    success: function (data) {

                    },
                    error: function (XMLHttpRequest, textStatus, errorThrown) {

                    },
                     complete: function () {
                         hidePreloader();
                    }
                });
            }
        }

Ação MVC:

[HttpPost]
[ValidateAntiForgeryToken] //<--- This is important
public async Task<IActionResult> StartWorkflow(IEnumerable<WorkflowModel> model)

E MODEL em C #:

public class WorkflowModel
 {
        public int OrderId { get; set; }
        public int OrderLineId { get; set; }
        public bool IsCustomOrderLine { get; set; }
        public string FolderPath { get; set; }
        public string FileName { get; set; }
 }

CONCLUSÃO:

O motivo do ERRO:

"Failed to load resource: the server responded with a status of 400 (Bad Request)"

É atributo: [ValidateAntiForgeryToken]para a ação MVCStartWorkflow

Solução em chamada Ajax:

  beforeSend: function (xhr) {//<--- This is important
                      xhr.setRequestHeader("RequestVerificationToken",
                      $('input:hidden[name="__RequestVerificationToken"]').val());
                    },

Para enviar a Lista de objetos você precisa formar dados como no exemplo (preenchendo o objeto da lista) e:

dados: {modelo: lista},

tipo: "post",

cócegas
fonte
0

É assim que funciona bem para mim:

var things = [
    { id: 1, color: 'yellow' },
    { id: 2, color: 'blue' },
    { id: 3, color: 'red' }
];

$.ajax({
    ContentType: 'application/json; charset=utf-8',
    dataType: 'json',
    type: 'POST',
    url: '/Controller/action',
    data: { "things": things },
    success: function () {
        $('#result').html('"PassThings()" successfully called.');
    },
    error: function (response) {
        $('#result').html(response);
    }
});

Com "ContentType" em "C" maiúsculo.

Mixomatose
fonte