The ASP.NET MVC controller action methods are primarily used for handling 'business' operations but it can be used for lots more.
I thought it would be fun to see what creative, useful things people have created actions for that may be practical or useful for others.
Here's my contribution :
Javascript file concatenator - to reduce number of http requests:
[OutputCache(Duration = 5 * 60, VaryByParam="")] // DONT USE "None" here *
public ContentResult RenderJavascript(){
StringBuilder js = new StringBuilder();
StringWriter sw = new StringWriter(js);
// load all my javascript files
js.AppendLine(File.ReadAllText(Request.MapPath("~/Scripts/jquery.hoverIntent.minified.js")));
js.AppendLine(File.ReadAllText(Request.MapPath("~/Scripts/jquery.corner.js")));
js.AppendLine(File.ReadAllText(Request.MapPath("~/Scripts/rollingrazor.js")));
return new ContentResult()
{
Content = js.ToString(),
ContentType = "application/x-javascript"
};
}
Map a route to it :
// javascript
routes.MapRoute(
"js-route",
"dynamic/js",
new { controller = "Application", action = "RenderJavascript" }
);
Refer to it from your master page :
<script type="text/javascript" src="/dynamic/js"></script>
Be warned I've set a cache for the output, so if you're changing your JS and refreshing the page you might want to disable the cache!
I jsut need to come back and figure out how to gzip it.
*
You shouldn't use VaryByParam="None" because that causes the Vary header to be send, which causes the browser to go back and check for a new version. If you really have to change your js content then your users are just goin to have to wait 5 minutes for it!
Does a HTTP 301 Redirect count as clever?
public class PermanentRedirectResult : ActionResult
{
public string Url { get; set; }
public PermanentRedirectResult(string url)
{
if (string.IsNullOrEmpty(url))
{
throw new ArgumentException("url is null or empty", "url");
}
this.Url = url;
}
public override void ExecuteResult(ControllerContext context)
{
if (context == null)
{
throw new ArgumentNullException("context");
}
context.HttpContext.Response.StatusCode = 301;
context.HttpContext.Response.RedirectLocation = Url;
}
}
View result with email confirmation:
public abstract class ViewResultWithConfirmationEmail: ViewResult
{
protected readonly string toAddress;
protected ViewResultWithConfirmationEmail(string toAddress)
{
this.toAddress = toAddress;
}
protected abstract MailMessage CreateEmail(ControllerContext context);
protected override void ExecuteResult(ControllerContext context)
{
MailMessage message = CreateEmail(context);
var smtpClient = new SmtpClient();
smtpClient.Send(message);
base.ExecuteResult(context);
}
}
Some of the implementation details have been omitted here, but I could use this to implement a RegistrationSuccessResult
class, for example, that would send an email message with the appropriate text after a successful user registration.
The reason I chose to send the message in a subclassed ViewResult instead of in the action method itself was to make it easier to separate my unit tests.
A partial implementation of less, a css syntax extender
It actually support only constants and what they called mixins, the source are here.
In this post I explained how to use it in mvc (the post is in italian but just look at the source):
you can just look
here at the result
Actually, not an "action" method, but a custom controller that implements a route-base RPC implementation. It identifies the contract and method from the route-data and passes the call onto the server-side service implementation. Pretty efficient (and yes, there are valid reasons that I'm not just using WCF or SOAP).
Not sure how many of you are using areas but an XCopy script is pretty necessary when making View changes in an Area. Areas compile into the parent project but only on compile (not on save) so to keep the dev agility, you'll need to run an XCopy.