JSON Web Services really are the jumping off point between server-centric and client-centric web development. You can add all of the pretty, JavaScript-y, AJAX-y sugar you want to your UI, but if your page or control posts back, (weather or not it is asynchronously via an Update Panel) then it is still server-centric in terms of logic.
But by off-loading your logic to JSON Web Services, you are making your pages themselves each act like a nice client-centric MVC application. The HTML is obviously the view, the web services act as the model, since they will be processing business objects for persistence or returning them to the UI for consumption. Finally, a JavaScript file sits in between, acting as a controller.
Model - Web Service
using System;
using System.Web.Services;
using System.Web.Script.Services;
namespace MyWebApplication
{
[WebService(Namespace = "http://tempuri.org/")]
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
[ScriptService]
public class JSONTestService : WebService
{
[WebMethod]
public void SetNewTest(Guid id, string text)
{
this.Application.Add(id.ToString(), text);
}
[WebMethod]
public string GetNewTest(Guid id)
{
return this.Application[id.ToString()].ToString();
}
[WebMethod]
public Guid CreateNew()
{
return Guid.NewGuid();
}
}
}
But normally, these web services will be processing full business objects. What I do is actually keep a variable on the client that stores my object, manipulate its properties via client-side events, (both of which are in the JavaScript file), and then pass the whole object to a web service for server-side manipulation.
View - Page / Control
<script type="text/javascript" language="javascript">
var txtGetNewTestID = '<% =this.txtGetNewTest.ClientID %>';
var txtSetNewTestID = '<% =this.txtSetNewTest.ClientID %>';
</script>
<table>
<tr>
<td>
</td>
<td>
<asp:Button id="btnNew" runat="server" Text="New" OnClientClick="GetNew(); return false;" />
</td>
</tr>
<tr>
<td>
<asp:TextBox ID="txtSetNewTest" runat="server" />
</td>
<td>
<asp:Button ID="btnSetNewTest" runat="server" Text="Set" OnClientClick="SetNewTest(); return false;" />
</td>
</tr>
<tr>
<td>
<asp:TextBox ID="txtGetNewTest" runat="server" />
</td>
<td>
<asp:Button ID="btnGet" runat="server" Text="Get" OnClientClick="GetNewTest(); return false;" />
</td>
</tr>
</table>
The only embedded JavaScript here is for variable declarations. Loose JavaScript files can't use the <% ... %> construct to call into .NET code, since they are not in the context of a page or control. So I like to expose the controls I'll need to manipulate this way, and then use $get(<control unique id>) in my JavaScript file to get a reference to the control.
You can still use server controls, and take advantage of the extensions they add to HTML controls. However, we don't want to cause a post back, as that will kill the state of our client objects. So for controls that always do an AutoPostBack, (like buttons) add "return false;" to end of the "client" event; this will not call __doPostback() behind the scenes.
Controller - JavaScript file
var _guid = null;
function GetNew()
{
MyWebApplication.JSONTestService.CreateNew(GetNewDone, OnError, null);
}
function GetNewDone(result)
{
_guid = result;
}
function SetNewTest()
{
MyWebApplication.JSONTestService.SetNewTest(_guid, $get(txtSetNewTestID).value, null, OnError, null);
}
function GetNewTest()
{
MyWebApplication.JSONTestService.GetNewTest(_guid, GetNewTestDone, OnError, null);
}
function GetNewTestDone(result)
{
$get(txtGetNewTestID).value = result;
}
function OnError(ex)
{
alert('Error: ' + ex._message);
}
If any of the server code in a web method on your web service throws an exception, the OnError JavaScript function here will be called. The variable passed is a JSON serialized Exception object, providing the description of the error, as well as a full stack trace! This way, you can still catch-and-log-and-rethrow your exceptions on the server, and gracefully deal with them on the client.
Another benefit to using a loose JavaScript file as the controller is that it provides a rich debugging experience. I wrote more about that here.
I know that using a JavaScript file as a controller in the MVC architecture is a stretch, but it works. Since you can swap .js files on the fly, they are kind of a configurable broker; as long as they handle events on the client and can, in response, call services on the server, the paradigm fits. This way, you can completely replace your back end, and then merely tweak your JavaScript file to keep communication between your persistence and your UI flowing.
The only thing left to do is wire everything up in the code-behind of the page / control:
using System;
using System.Web.UI;
namespace MyWebApplication
{
public partial class JSONTest : UserControl //or Page
{
protected void Page_Load(object sender, EventArgs e)
{
if (!this.Page.IsPostBack)
{
ScriptManager sm = ScriptManager.GetCurrent(this.Page);
sm.Scripts.Add(new ScriptReference("/JSONTest.js"));
sm.Services.Add(new ServiceReference("/JSONTestService.asmx"));
}
}
}
}
In fact, since I've adopted this architecture, the only code-behind my pages and controls have is in the Load event, if IsPostBack is false. This separation of logic from presentation makes for a much cleaner ASPX / ASCX file, both in terms of managed code behind and HTML!
Despite the fact that the implementation of the "model" web services is still .NET code, they are called asynchronously though the AJAX web service infrastructure, not via a form submit, and therefore do not cause the Load event of your page to fire. This really does away with the page lifecycle as we know it. This way, we can move all of the business logic from the page to JavaScript, and then use JSON Web Services to either do persistence and call other services, or do any processing that is too complex for, or not supported by, JavaScript (for example, guids).
Speaking of the implementation of these web methods, there is more thing to point out. In the first post I linked to in this article, I talk about one main "reality check" of JSON Web Services: they do not share the same Session as the UI. So my workaround for this is to use the Application object, and store objects in it keyed off of guids.
This way, although all of instances of our app are using the same Application object, there will be no name collisions. And as far as the server is concerned, and object in memory is an object in memory; there shouldn't be any performance ramifications using this method verses using Session state.
JSON from AJAX and ASP.NET
Hot on Web: