Jonn
Jonn

Reputation: 4661

Extending JsonResult

How do you extend JsonResult? Say I wanted to make a JsonTransactionResult because I want to enforce all of my transactions to return a jsonified TransactionResult object. The TransactionResult object contains data for error messages and stuff. Do I do this via inheritance or wrapping JsonResult?

Upvotes: 1

Views: 2614

Answers (1)

MartinHN
MartinHN

Reputation: 19782

I'd simply inherit from JsonResult and return an instance of the TransactionResult class.

I have something similar, though I inherit from ActionResult and use JSON.NET, since I had some serialization issues with DateTime using the built in JsonResult.

/// <summary>
/// A Newtonsoft.Json based JsonResult for ASP.NET MVC
/// </summary>
public class JsonNetResult : ActionResult
{
    /// <summary>
    /// Initializes a new instance of the <see cref="JsonNetResult"/> class.
    /// </summary>
    public JsonNetResult()
    {
        this.SerializerSettings = new JsonSerializerSettings();
    }

    /// <summary>
    /// Gets or sets the content encoding.
    /// </summary>
    /// <value>The content encoding.</value>
    public Encoding ContentEncoding { get; set; }

    /// <summary>
    /// Gets or sets the type of the content.
    /// </summary>
    /// <value>The type of the content.</value>
    public string ContentType { get; set; }

    /// <summary>
    /// Gets or sets the data.
    /// </summary>
    /// <value>The data object.</value>
    public object Data { get; set; }

    /// <summary>
    /// Gets or sets the serializer settings.
    /// </summary>
    /// <value>The serializer settings.</value>
    public JsonSerializerSettings SerializerSettings { get; set; }

    /// <summary>
    /// Gets or sets the formatting.
    /// </summary>
    /// <value>The formatting.</value>
    public Formatting Formatting { get; set; }

    /// <summary>
    /// Enables processing of the result of an action method by a custom type that inherits from the <see cref="T:System.Web.Mvc.ActionResult"/> class.
    /// </summary>
    /// <param name="context">The context in which the result is executed. The context information includes the controller, HTTP content, request context, and route data.</param>
    public override void ExecuteResult(ControllerContext context)
    {
        if (context == null)
        {
            throw new ArgumentNullException("context");
        }

        HttpResponseBase response = context.HttpContext.Response;

        response.ContentType = !String.IsNullOrWhiteSpace(this.ContentType) ? this.ContentType : "application/json";

        if (this.ContentEncoding != null)
        {
            response.ContentEncoding = this.ContentEncoding;
        }

        if (this.Data != null)
        {
            JsonTextWriter writer = new JsonTextWriter(response.Output) { Formatting = this.Formatting };

            JsonSerializer serializer = JsonSerializer.Create(this.SerializerSettings);
            serializer.Serialize(writer, this.Data);

            writer.Flush();
        }
    }
}

And then I inherit from that class, to wrap a success property with the result:

/// <summary>
/// Derives from <see cref="JsonNetResult"/>. This action result can be used to wrap an AJAX callback result with a status code and a description, along with the actual data.
/// </summary>
public class CallbackJsonResult : JsonNetResult
{
    /// <summary>
    /// Initializes a new instance of the <see cref="CallbackJsonResult"/> class.
    /// </summary>
    /// <param name="statusCode">The status code.</param>
    public CallbackJsonResult(HttpStatusCode statusCode)
    {
        this.Initialize(statusCode, null, null);
    }

    /// <summary>
    /// Initializes a new instance of the <see cref="CallbackJsonResult"/> class.
    /// </summary>
    /// <param name="statusCode">The status code.</param>
    /// <param name="description">The description.</param>
    public CallbackJsonResult(HttpStatusCode statusCode, string description)
    {
        this.Initialize(statusCode, description, null);
    }

    /// <summary>
    /// Initializes a new instance of the <see cref="CallbackJsonResult"/> class.
    /// </summary>
    /// <param name="statusCode">The status code.</param>
    /// <param name="data">The callback result data.</param>
    public CallbackJsonResult(HttpStatusCode statusCode, object data)
    {
        this.Initialize(statusCode, null, data);
    }

    /// <summary>
    /// Initializes a new instance of the <see cref="CallbackJsonResult"/> class.
    /// </summary>
    /// <param name="statusCode">The status code.</param>
    /// <param name="description">The description.</param>
    /// <param name="data">The callback result data.</param>
    public CallbackJsonResult(HttpStatusCode statusCode, string description, object data)
    {
        this.Initialize(statusCode, description, data);
    }

    /// <summary>
    /// Initializes this instance.
    /// </summary>
    /// <param name="statusCode">The status code.</param>
    /// <param name="description">The description.</param>
    /// <param name="data">The callback result data.</param>
    private void Initialize(HttpStatusCode statusCode, string description, object data)
    {
        Data = new { Success = statusCode == HttpStatusCode.OK, Status = (int)statusCode, Description = description, Data = data };
    }
}

Upvotes: 3

Related Questions