<?xml version="1.0" encoding="utf-8"?>
<rss xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:pingback="http://madskills.com/public/xml/rss/module/pingback/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/" xmlns:dc="http://purl.org/dc/elements/1.1/" version="2.0">
  <channel>
    <title>Jay Harris is Cpt. LoadTest - Learn to Code</title>
    <link>http://www.cptloadtest.com/</link>
    <description>a .net developers blog on improving user experience of humans and coders</description>
    <language>en-us</language>
    <copyright>Jason Harris</copyright>
    <lastBuildDate>Wed, 07 Apr 2010 15:27:53 GMT</lastBuildDate>
    <generator>newtelligence dasBlog 2.3.12105.0</generator>
    <managingEditor>jharris@harrisdesigns.com</managingEditor>
    <webMaster>jharris@harrisdesigns.com</webMaster>
    <item>
      <trackback:ping>http://www.cptloadtest.com/Trackback.aspx?guid=5a613524-5607-4fbb-a89f-74b5afb7c5e9</trackback:ping>
      <pingback:server>http://www.cptloadtest.com/pingback.aspx</pingback:server>
      <pingback:target>http://www.cptloadtest.com/PermaLink,guid,5a613524-5607-4fbb-a89f-74b5afb7c5e9.aspx</pingback:target>
      <dc:creator>Jay Harris</dc:creator>
      <wfw:comment>http://www.cptloadtest.com/CommentView,guid,5a613524-5607-4fbb-a89f-74b5afb7c5e9.aspx</wfw:comment>
      <wfw:commentRss>http://www.cptloadtest.com/SyndicationService.asmx/GetEntryCommentsRss?guid=5a613524-5607-4fbb-a89f-74b5afb7c5e9</wfw:commentRss>
      <body xmlns="http://www.w3.org/1999/xhtml">
        <p>
Aligned with another jam session at Ann Arbor's <a href="http://www.comejamwithus.org">Come
Jam With Us</a> is another installment of Learn to Code, this time providing an introduction
to WatiN, or Web Application Testing in .NET. The jam session was held at the offices
of SRT Solutions in Ann Arbor, Michigan, at 5:30p, Tuesday April 6th. Though thunderstorms
were in the forecast, the predicted high was 72°F (22°C), so we weren't bothered by
the same 8" of fluffy white snow that caused cancellations and delays during
the my session on ASP.NET MVC 2. But for those that couldn't make the WatiN jam session,
might I recommend the exercise below.
</p>
        <h3>About This Exercise
</h3>
        <p>
This coding exercise is designed to give you an introduction to browser-based testing
using the WatiN framework, or Web Application Testing in .NET. The framework allows
developers to create integration tests (using a unit testing framework like MbUnit,
NUnit, or MSTest) to test and assert their application within a browser window. The
framework interacts with the browser DOM much like and end-user, producing reliable
results that mimic the real world. In this sample, we will write a few WatiN tests
against the Google search engine.
</p>
        <h3>Prerequisites
</h3>
        <p>
To complete this exercise, you will need to meet or complete a few prerequisites.
Please complete these prerequisites before moving on. The session is designed to be
completed in about an hour, but setup and prerequisites are not included in that time.
</p>
        <ul>
          <li>
An active internet connection. (Our tests will be conducted against live third-party
sites.) 
</li>
          <li>
Install Microsoft Visual Studio 2008 or Microsoft Visual Studio 2010. 
</li>
          <li>
            <a title="Download WATiN from SourceForge" href="http://sourceforge.net/projects/watin/files/">Download</a> and
extract the latest version of the WatiN framework. 
</li>
        </ul>
        <h3>Exercise 0: Getting Started
</h3>
        <h4>Creating a Project
</h4>
        <p>
WatiN is generally used within the context of a unit testing framework. For this exercise,
we will be using a Visual Studio Test Project and MSTest to wrap our WatiN code.
</p>
        <ol>
          <li>
Create a new "Test Project" in Visual Studio named "WatinSample".
The language is up to you, but all of the examples in this post will use C#. 
</li>
          <li>
Feel free to delete the Authoring Tests document, the Manual Test file, and UnitTest1.cs.
We won't be using these. 
</li>
          <li>
Add a reference to WatiN.Core.dll from the bin directory of your extracted WatiN download. 
</li>
          <li>
Compile. 
</li>
        </ol>
        <h3>Exercise 1: My First Browser Tests
</h3>
        <p>
In our first test, we will use the project we just created to test Google's home page.
After accessing <a href="http://www.google.com">http://www.google.com</a>, we will
check a few properties of the browser and a few loaded elements to ensure that the
expected page was returned. The first thing we will need is a new Unit Test class
to start our testing.
</p>
        <ol>
          <li>
Create a new class (Right click on the "WatinSample" project and select
Add –&gt; Class…), called WhenViewingTheGoogleHomePage. 
</li>
          <li>
Mark the class as public. 
</li>
          <li>
Add the MSTest [TestClass] attribute to the new class. 
</li>
          <li>
Compile. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">using Microsoft.VisualStudio.TestTools.UnitTesting;

namespace WatinSample
{
  [TestClass]
  public class WhenViewingTheGoogleHomePage
  {
  }
}</pre>
        <h4>Make an instance of the browser
</h4>
        <p>
Now that we have a test class, we can start writing WatiN code. Each of our tests
will first need a Browser object to test against. Using methods attributed with TestInitialize
and TestCleanup, we can create a browser instance before the test starts and shut
it down when the test is complete.
</p>
        <p>
Creating an instance of a browser in WatiN is easy: simply create a new instance of
the IE class, passing in a URL. We can assign this new class to a field of type Browser,
which is a base class of all browser classes in WatiN. Currently, WatiN supports Internet
Explorer and Firefox.
</p>
        <ol>
          <li>
Create a private field in the test class named browserInstance of type WatiN.Core.Browser.
Add a using statement to WatiN.Core if you wish. 
</li>
          <li>
Create a test initialization method named WithAnInstanceOfTheBrowser and give it the
[TestInitialize] attribute. Within this method, create a new instance of the IE class,
passing in the Google URL, <a href="http://www.google.com">http://www.google.com</a>,
and assigning the instance to the browserInstance field. 
</li>
          <li>
Finally, create a test cleanup method named ShutdownBrowserWhenDone and give it the
[TestCleanup] attribute. Within this method, execute the Close() method on our browser
instance and assign the field to null to assist with object disposal. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">using Microsoft.VisualStudio.TestTools.UnitTesting;
using WatiN.Core;

namespace WatinSample
{
  [TestClass]
  public class WhenViewingTheGoogleHomePage
  {
    Browser browserInstance;

    [TestInitialize]
    public void WithAnInstanceOfTheBrowser()
    {
      browserInstance = new IE("http://www.google.com");
    }

    [TestCleanup]
    public void ShutdownBrowserWhenDone()
    {
      browserInstance.Close();
      browserInstance = null;
    }
  }
}</pre>
        <h4>Our First Tests: Checking for existence of an element
</h4>
        <p>
There are three prominent items on the Google home page: the Google logo, the search
criteria text box, and the search button. Using WatiN, we can check for them all.
The WatiN Browser object contains an Elements collection, which is a flattened collection
of every element in the entire DOM. Like any collection, you can use Linq and lambda
expressions to search for items within this collection. Alternately, you may also
use the Element method, which accepts the same lambda expression that would be used
within the Where extension method on the collection, and returns the first or default
element. For more specific searches, WatiN's Browser object includes similar collections
and methods for searching explicitly for Images (&lt;IMG&gt;), Paras (&lt;P&gt;),
Text Fields (&lt;INPUT type="text" /&gt;), and so on.
</p>
        <p>
On each returned Element (or derived Para, Image, or Text Field, etc., all of which
inherit from Element), WatiN supplies properties for accessing the CSS Class, Id,
InnerHtml, Name, Tag, Text, Value, or many other attributes. The method GetAttributeValue(string
attributeName) is provided for accessing other attributes that are not explicitly
defined on the object (uncommon attributes and custom attributes). Finally, elements
also contain a Style property, which not only gives access to the inline style attribute,
but also any CSS properties associated with the element from Internal Style (in the
Page Head) or External Style (in an external style sheet).
</p>
        <p>
On to checking for the three elements within the Google home page: the logo, the criteria
input, and the search button. First, check for the existence of the Google logo graphic.
The image can be found by searching the DOM for an image with an Id of "logo".
WatiN works very closely with lambda expressions, so we can use these to help us find
out graphic.
</p>
        <ol>
          <li>
Create a new public method named PageShouldContainGoogleLogo. 
</li>
          <li>
Add the MSTest [TestMethod] attribute to the method. 
</li>
          <li>
Search for and assert on the existence of an image with the Id of "logo". 
</li>
          <li>
Optionally, we can also check that the image has the expected Alt attribute; in this
case, the value should be "Google". 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void PageShouldContainGoogleLogo()
{
  Image googleLogo;
  googleLogo = browserInstance.Image(img =&gt; img.Id == "logo");
  Assert.IsTrue(googleLogo.Exists);
  Assert.AreEqual("Google", googleLogo.Alt);
}</pre>
        <p>
Next, check for the existence of the search criteria input box. WatiN refers to these
elements as Text Fields, using the TextField type. Additionally, this form field is
identified by its Name rather than its Id. In Google, the name given to the criteria
input is "q".
</p>
        <ol>
          <li>
Create a new public method named PageShouldContainSearchCriteriaInput and give it
the [TestMethod] attribute. 
</li>
          <li>
Search for and assert on the existence of a Text Field with the name "q". 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void PageShouldContainSearchCriteriaInput()
{
  TextField criteriaInput;
  criteriaInput = browserInstance.TextField(tf =&gt; tf.Name == "q");
  Assert.IsTrue(criteriaInput.Exists);
}</pre>
        <p>
Finally, check for the existence of the search button using the Button method. In
our lambda expression, it is not important to know if the field is identified by a
Name property or an Id attribute, as WatiN supplies a IdOrName property to help us
find the element. The value to identify the button is "btnG".
</p>
        <ol>
          <li>
Create a new public method named PageShouldContainSearchButton and give it the [TestMethod]
attribute. 
</li>
          <li>
Search for and assert on the existence of a Button with the Id or Name of 'btnG". 
</li>
          <li>
Optionally, we can also check the value of the button, which is the text displayed
on the button on-screen. This text should be "Google Search". 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void PageShouldContainSearchButton()
{
  Button searchButton;
  searchButton = browserInstance.Button(btn =&gt; btn.IdOrName == "btnG");
  Assert.IsTrue(searchButton.Exists);
  Assert.AreEqual("Google Search", searchButton.Value);
}</pre>
        <h4>Working with Style
</h4>
        <p>
WatiN can access properties on the DOM beyond just Text values and Alt attributes.
WatiN also has full access to the style that CSS has applied to an element. Let's
check out a few CSS properties, both those explicitly defined by WatiN and those implicitly
accessible through the WatiN framework.
</p>
        <p>
For our first style check, we'll take a look at the default font family used on the
Google Home Page. Font Family is one of the explicitly available style properties
on a WatiN element. Some others, like Color, Display, and Height are also explicitly
defined.
</p>
        <ol>
          <li>
Create a new public test method named BodyShouldUseArialFontFamily. 
</li>
          <li>
Assert that the font family assigned to the body matches "arial, sans-serif". 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void BodyShouldUseArialFontFamily()
{
  Assert.AreEqual("arial, sans-serif", browserInstance.Body.Style.FontFamily);
}</pre>
        <p>
For our second style check, we will look for an implicit style definition. At the
top of the Google Home Page is a series of links to other areas of Google, such as
Images, Videos, Maps, and News. At the end of this list is a More link, that when
clicked, displays a hidden DIV tag containing even more links, such as Books, Finance,
and Google Translate. Since we do not have any code in our test initialization that
interacts with the browser, and thus nothing that is clicking the More link, that
DIV should still have a hidden visibility. However, since Visibility isn't an explicitly
defined style property within WatiN, we need to use the GetAttributeValue method to
retrieve the current visibility setting.
</p>
        <ol>
          <li>
Create a new public test method named MoreItemsShouldNotBeVisibleOnPageLoad. 
</li>
          <li>
Search for the More Items DIV. It's Id is "gbi". 
</li>
          <li>
Using the property lookup method, GetAttributeValue(string attributeName), check that
the Visibility is set to "hidden". 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void MoreItemsShouldNotBeVisibleOnPageLoad()
{
  var googleBarMoreItems = browserInstance.Div(gbi =&gt; gbi.Id == "gbi");
  Assert.AreEqual("hidden", googleBarMoreItems.Style.GetAttributeValue("visibility"));
}</pre>
        <h3>Exercise 2: Interacting with the Browser
</h3>
        <p>
Browser Integration tests are more than just loading a page and checking a few element
attributes. Our tests may also need to enter values into form fields, click links
and buttons, or interact with browser navigation like the back button. WatiN fully
supports all of these features in a very intuitive fashion.
</p>
        <h4>A new test class, this time with Search Capability
</h4>
        <p>
Create a new test class, similar to what we did in Exercise 1, calling the new test
class WhenViewingGoogleSearchResultsForComeJamWithUs. Also add in the TestInitialize
and TestCleanup methods that open and close the browser. However, this time, after
we load <a href="http://www.google.com">http://www.google.com</a>, enter a value into
the search criteria input and then click the Google Search button.
</p>
        <ol>
          <li>
Create a new class named WhenViewingGoogleSearchResultsForComeJamWithUs, similar to
what was done in Exercise 1. 
</li>
          <li>
Add in the TestInitialize and TestCleanup methods from Exercise 1. Name the Initialize
method WithAnInstanceOfTheBrowserSearchingGoogle. 
</li>
          <li>
After the code that initializes the IE class, find the search criteria Text Field
and set its value to "Come Jam With Us". 
</li>
          <li>
After setting the Text Field value, click the Google Search button by calling the
Click() method on the Button class. 
</li>
          <li>
Compile. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">using Microsoft.VisualStudio.TestTools.UnitTesting;
using WatiN.Core;

namespace WatinSample
{
  [TestClass]
  public class WhenViewingGoogleSearchResultsForComeJamWithUs
  {
    Browser browserInstance;

    [TestInitialize]
    public void WithAnInstanceOfTheBrowserSearchingGoogle()
    {
      browserInstance = new IE(@"http://www.google.com");
      TextField criteria =
        browserInstance.TextField(tf =&gt; tf.Name == "q");
      criteria.Value = "Come Jam With Us";
      Button search =
        browserInstance.Button(btn =&gt; btn.IdOrName == "btnG");
      search.Click();
    }

    [TestCleanup]
    public void ShutdownBrowserWhenDone()
    {
      browserInstance.Close();
      browserInstance = null;
    }
  }
}</pre>
        <p>
With this code, or initialized test will load the Google Home Page and will conduct
a search for "Come Jam With Us".
</p>
        <h4>Validating the Search Results Page
</h4>
        <p>
For our first verification, let's check the URL for the browser window. The search
result URL should contain the search criteria in the URL's query string; we can validate
this using the URL property on our instance of the Browser object.
</p>
        <ol>
          <li>
Create a new public test method named BrowserUrlShouldContainSearchCriteria. 
</li>
          <li>
Validate that the current browser URL contains the search criteria information, "q=Come+Jam+With+Us". 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void BrowserUrlShouldContainSearchCriteria()
{
  Assert.IsTrue(browserInstance.Url.Contains(@"q=Come+Jam+With+Us"));
}</pre>
        <h4>Finding Child Elements
</h4>
        <p>
With WatiN, we are not just limited to searching for items directly from the Browser
object. We can also search for child elements directly from their parent element or
any ancestor element. Our search results should contain a search result item linking
to the Come Jam With Us web site. The Google Results page contains a DIV identified
as "res" that serves as a container for all search result information. Rather
than checking that our Come Jam With Us link exists somewhere on the page, we should
search for it directly within the results DIV.
</p>
        <ol>
          <li>
Create a new public test method named ResultsShouldContainLinkToComeJamWithUs. 
</li>
          <li>
From the browser instance, find a DIV identified as "res". 
</li>
          <li>
Assert that a link to <a href="http://www.comejamwithus.org">http://www.comejamwithus.org</a> exists
within the "res" DIV. 
</li>
          <li>
Compile and run the test. The test should pass. 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void ResultsShouldContainLinkToComeJamWithUs()
{
  Link comeJamWithUs;
  Div searchResults = browserInstance.Div(div =&gt; div.IdOrName == "res");
  comeJamWithUs =
    searchResults.Link(link =&gt; link.Url == @"http://www.comejamwithus.org/");
  Assert.IsTrue(comeJamWithUs.Exists);
}</pre>
        <h4>Inner Text verses InnerHtml
</h4>
        <p>
An element may contain many child elements. An anchor tag—&lt;A href="#"&gt;—can
contain text, and child elements may make portions of that text bold, italic, underlined,
or even bright red. Through WatiN, we can access that inner content as straight text
without the formatting, or as the InnerHtml including all of the child elements.
</p>
        <ol>
          <li>
Create two public test methods, one named ResultsLinkContainsComeJamWithUsText and
the other named ResultsLinkContainsComeJamWithUsHtml. 
</li>
          <li>
In both methods, search for the results DIV, as we did in the previous test. 
</li>
          <li>
In both methods, search through the results DIV for a link with a URL matching <a href="http://www.comejamwithus.org">http://www.comejamwithus.org</a></li>
          <li>
In the Text method, assert that the text of the link matches "Come Jam with us
(Software Development Study Group)". Note that the value contains no child HTML
elements. 
</li>
          <li>
In the HTML method, assert that the InnerHtml of the link matches "&lt;EM&gt;Come
Jam with us&lt;/EM&gt; (Software Development Study Group)". Note that for the
same link, we now have the emphasis tags surrounding Come Jam With Us. 
</li>
          <li>
Compile and run both tests. The tests should pass.</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void ResultsLinkContainsComeJamWithUsText()
{
  Link comeJamWithUs;
  Div searchResults = browserInstance.Div(div =&gt; div.IdOrName == "res");
  comeJamWithUs =
    searchResults.Link(link =&gt; link.Url == @"http://www.comejamwithus.org/");
  Assert.AreEqual(@"Come Jam with us (Software Development Study Group)",
    comeJamWithUs.Text);
}

[TestMethod]
public void ResultsLinkContainsComeJamWithUsHtml()
{
  Link comeJamWithUs;
  Div searchResults = browserInstance.Div(div =&gt; div.IdOrName == "res");
  comeJamWithUs =
    searchResults.Link(link =&gt; link.Url == @"http://www.comejamwithus.org/");
  Assert.AreEqual(
    @"&lt;EM&gt;Come Jam with us&lt;/EM&gt; (Software Development Study Group)",
    comeJamWithUs.InnerHtml);
}</pre>
        <h4>Back to the Start
</h4>
        <p>
As previously mentioned, we can also fully interact with the browser, itself. Our
test initialization started from the Google Home Page and performed a search. Using
functionality built in to WatiN, we can execute the browser's back navigation to return
to the previous page.
</p>
        <p>
For our next test, execute a back navigation and verify that the browser's URL matches <a href="http://www.google.com/">http://www.google.com/</a>.
</p>
        <ol>
          <li>
Create a public test method named PageShouldHaveComeFromGoogleDotCom. 
</li>
          <li>
Execute back navigation in the browser by calling the Back() method on browserInstance. 
</li>
          <li>
Validate that the browser URL matches <a href="http://www.google.com/">http://www.google.com/</a>. 
</li>
          <li>
Compile and run the test. The test should pass.</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void PageShouldHaveComeFromGoogleDotCom()
{
  string previousUrl;
  browserInstance.Back();
  previousUrl = browserInstance.Url;
  Assert.AreEqual(@"http://www.google.com/", previousUrl);
}</pre>
        <h4>Putting it all together
</h4>
        <p>
Some interactions on a page cause element properties to change. An example of this
is the More link from Exercise 1; when the end-user clicks the More link, the More
Items DIV appears because the link's click event changes the Visibility style property
of the DIV to visible. For our final test, we will use what we have learned to test
this functionality.
</p>
        <ol>
          <li>
Create a new public test method named MoreItemsShouldBeVisibleOnMoreLinkClick.</li>
          <li>
Search for the header bar of Google links, a DIV with an Id of "gbar".</li>
          <li>
Within "gbar", search for the More Items DIV by an Id or Name of "gbi". 
</li>
          <li>
Assert that the visibility style property has a value of "hidden". 
</li>
          <li>
Within "gbar", search for the More link by its class name, "gb3".
Note that since a class attribute may contain multiple class definitions, this is
accomplished by validating that the class attribute contains the class you are searching
for. 
</li>
          <li>
Execute a Click event on the link. 
</li>
          <li>
Assert that the visibility style property of the More Items DIV has changed to "visible". 
</li>
        </ol>
        <pre class="csharp:nocontrols" name="code">[TestMethod]
public void MoreItemsShouldBeVisibleOnMoreLinkClick()
{
  var googleBar = browserInstance.Div(gbar =&gt; gbar.Id == "gbar");
  var googleBarMoreItems = googleBar.Div(gbi =&gt; gbi.Id == "gbi");
  Assert.AreEqual("hidden",
    googleBarMoreItems.Style.GetAttributeValue("visibility"));
  var googleBarMoreLink =
    googleBar.Link(link =&gt; link.ClassName.Contains("gb3"));
  googleBarMoreLink.Click();
  Assert.AreEqual("visible",
    googleBarMoreItems.Style.GetAttributeValue("visibility"));
}</pre>
        <h4>That's It
</h4>
        <p>
Now that we have spent some time on basic properties, interactions, and style sheets
within the WatiN framework, hopefully you can apply this to your own application and
get started with your own browser-based integration tests. If you would like more
information, I encourage you to check out the WatiN site at <a href="http://watin.sourceforge.net">http://watin.sourceforge.net</a>.
And as always, if you have any questions, drop me a line.
</p>
        <div style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; display: inline; float: none; padding-top: 0px" id="scid:0767317B-992E-4b12-91E0-4F059A8CECA8:9a272a14-7cc1-4234-a4bc-8531ce7b8597" class="wlWriterEditableSmartContent">Technorati
Tags: <a href="http://technorati.com/tags/WatiN" rel="tag">WatiN</a>,<a href="http://technorati.com/tags/Jam+Session" rel="tag">Jam
Session</a>,<a href="http://technorati.com/tags/Coding+Exercise" rel="tag">Coding
Exercise</a>,<a href="http://technorati.com/tags/Learn+to+Code" rel="tag">Learn to
Code</a>,<a href="http://technorati.com/tags/Browser+Testing" rel="tag">Browser Testing</a></div>
        <img width="0" height="0" src="http://www.cptloadtest.com/aggbug.ashx?id=5a613524-5607-4fbb-a89f-74b5afb7c5e9" />
      </body>
      <title>Learn to Code WatiN: Browser Test your Web Site with WatiN</title>
      <guid isPermaLink="false">http://www.cptloadtest.com/PermaLink,guid,5a613524-5607-4fbb-a89f-74b5afb7c5e9.aspx</guid>
      <link>http://www.cptloadtest.com/2010/04/07/Learn-To-Code-WatiN-Browser-Test-Your-Web-Site-With-WatiN.aspx</link>
      <pubDate>Wed, 07 Apr 2010 15:27:53 GMT</pubDate>
      <description>&lt;p&gt;
Aligned with another jam session at Ann Arbor's &lt;a href="http://www.comejamwithus.org"&gt;Come
Jam With Us&lt;/a&gt; is another installment of Learn to Code, this time providing an introduction
to WatiN, or Web Application Testing in .NET. The jam session was held at the offices
of SRT Solutions in Ann Arbor, Michigan, at 5:30p, Tuesday April 6th. Though thunderstorms
were in the forecast, the predicted high was 72°F (22°C), so we weren't bothered by
the same 8&amp;quot; of fluffy white snow that caused cancellations and delays during
the my session on ASP.NET MVC 2. But for those that couldn't make the WatiN jam session,
might I recommend the exercise below.
&lt;/p&gt;
&lt;h3&gt;About This Exercise
&lt;/h3&gt;
&lt;p&gt;
This coding exercise is designed to give you an introduction to browser-based testing
using the WatiN framework, or Web Application Testing in .NET. The framework allows
developers to create integration tests (using a unit testing framework like MbUnit,
NUnit, or MSTest) to test and assert their application within a browser window. The
framework interacts with the browser DOM much like and end-user, producing reliable
results that mimic the real world. In this sample, we will write a few WatiN tests
against the Google search engine.
&lt;/p&gt;
&lt;h3&gt;Prerequisites
&lt;/h3&gt;
&lt;p&gt;
To complete this exercise, you will need to meet or complete a few prerequisites.
Please complete these prerequisites before moving on. The session is designed to be
completed in about an hour, but setup and prerequisites are not included in that time.
&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;
An active internet connection. (Our tests will be conducted against live third-party
sites.) 
&lt;/li&gt;
&lt;li&gt;
Install Microsoft Visual Studio 2008 or Microsoft Visual Studio 2010. 
&lt;/li&gt;
&lt;li&gt;
&lt;a title="Download WATiN from SourceForge" href="http://sourceforge.net/projects/watin/files/"&gt;Download&lt;/a&gt; and
extract the latest version of the WatiN framework. 
&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Exercise 0: Getting Started
&lt;/h3&gt;
&lt;h4&gt;Creating a Project
&lt;/h4&gt;
&lt;p&gt;
WatiN is generally used within the context of a unit testing framework. For this exercise,
we will be using a Visual Studio Test Project and MSTest to wrap our WatiN code.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new &amp;quot;Test Project&amp;quot; in Visual Studio named &amp;quot;WatinSample&amp;quot;.
The language is up to you, but all of the examples in this post will use C#. 
&lt;/li&gt;
&lt;li&gt;
Feel free to delete the Authoring Tests document, the Manual Test file, and UnitTest1.cs.
We won't be using these. 
&lt;/li&gt;
&lt;li&gt;
Add a reference to WatiN.Core.dll from the bin directory of your extracted WatiN download. 
&lt;/li&gt;
&lt;li&gt;
Compile. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;h3&gt;Exercise 1: My First Browser Tests
&lt;/h3&gt;
&lt;p&gt;
In our first test, we will use the project we just created to test Google's home page.
After accessing &lt;a href="http://www.google.com"&gt;http://www.google.com&lt;/a&gt;, we will
check a few properties of the browser and a few loaded elements to ensure that the
expected page was returned. The first thing we will need is a new Unit Test class
to start our testing.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new class (Right click on the &amp;quot;WatinSample&amp;quot; project and select
Add –&amp;gt; Class…), called WhenViewingTheGoogleHomePage. 
&lt;/li&gt;
&lt;li&gt;
Mark the class as public. 
&lt;/li&gt;
&lt;li&gt;
Add the MSTest [TestClass] attribute to the new class. 
&lt;/li&gt;
&lt;li&gt;
Compile. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;using Microsoft.VisualStudio.TestTools.UnitTesting;

namespace WatinSample
{
  [TestClass]
  public class WhenViewingTheGoogleHomePage
  {
  }
}&lt;/pre&gt;
&lt;h4&gt;Make an instance of the browser
&lt;/h4&gt;
&lt;p&gt;
Now that we have a test class, we can start writing WatiN code. Each of our tests
will first need a Browser object to test against. Using methods attributed with TestInitialize
and TestCleanup, we can create a browser instance before the test starts and shut
it down when the test is complete.
&lt;/p&gt;
&lt;p&gt;
Creating an instance of a browser in WatiN is easy: simply create a new instance of
the IE class, passing in a URL. We can assign this new class to a field of type Browser,
which is a base class of all browser classes in WatiN. Currently, WatiN supports Internet
Explorer and Firefox.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a private field in the test class named browserInstance of type WatiN.Core.Browser.
Add a using statement to WatiN.Core if you wish. 
&lt;/li&gt;
&lt;li&gt;
Create a test initialization method named WithAnInstanceOfTheBrowser and give it the
[TestInitialize] attribute. Within this method, create a new instance of the IE class,
passing in the Google URL, &lt;a href="http://www.google.com"&gt;http://www.google.com&lt;/a&gt;,
and assigning the instance to the browserInstance field. 
&lt;/li&gt;
&lt;li&gt;
Finally, create a test cleanup method named ShutdownBrowserWhenDone and give it the
[TestCleanup] attribute. Within this method, execute the Close() method on our browser
instance and assign the field to null to assist with object disposal. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;using Microsoft.VisualStudio.TestTools.UnitTesting;
using WatiN.Core;

namespace WatinSample
{
  [TestClass]
  public class WhenViewingTheGoogleHomePage
  {
    Browser browserInstance;

    [TestInitialize]
    public void WithAnInstanceOfTheBrowser()
    {
      browserInstance = new IE(&amp;quot;http://www.google.com&amp;quot;);
    }

    [TestCleanup]
    public void ShutdownBrowserWhenDone()
    {
      browserInstance.Close();
      browserInstance = null;
    }
  }
}&lt;/pre&gt;
&lt;h4&gt;Our First Tests: Checking for existence of an element
&lt;/h4&gt;
&lt;p&gt;
There are three prominent items on the Google home page: the Google logo, the search
criteria text box, and the search button. Using WatiN, we can check for them all.
The WatiN Browser object contains an Elements collection, which is a flattened collection
of every element in the entire DOM. Like any collection, you can use Linq and lambda
expressions to search for items within this collection. Alternately, you may also
use the Element method, which accepts the same lambda expression that would be used
within the Where extension method on the collection, and returns the first or default
element. For more specific searches, WatiN's Browser object includes similar collections
and methods for searching explicitly for Images (&amp;lt;IMG&amp;gt;), Paras (&amp;lt;P&amp;gt;),
Text Fields (&amp;lt;INPUT type=&amp;quot;text&amp;quot; /&amp;gt;), and so on.
&lt;/p&gt;
&lt;p&gt;
On each returned Element (or derived Para, Image, or Text Field, etc., all of which
inherit from Element), WatiN supplies properties for accessing the CSS Class, Id,
InnerHtml, Name, Tag, Text, Value, or many other attributes. The method GetAttributeValue(string
attributeName) is provided for accessing other attributes that are not explicitly
defined on the object (uncommon attributes and custom attributes). Finally, elements
also contain a Style property, which not only gives access to the inline style attribute,
but also any CSS properties associated with the element from Internal Style (in the
Page Head) or External Style (in an external style sheet).
&lt;/p&gt;
&lt;p&gt;
On to checking for the three elements within the Google home page: the logo, the criteria
input, and the search button. First, check for the existence of the Google logo graphic.
The image can be found by searching the DOM for an image with an Id of &amp;quot;logo&amp;quot;.
WatiN works very closely with lambda expressions, so we can use these to help us find
out graphic.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public method named PageShouldContainGoogleLogo. 
&lt;/li&gt;
&lt;li&gt;
Add the MSTest [TestMethod] attribute to the method. 
&lt;/li&gt;
&lt;li&gt;
Search for and assert on the existence of an image with the Id of &amp;quot;logo&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Optionally, we can also check that the image has the expected Alt attribute; in this
case, the value should be &amp;quot;Google&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void PageShouldContainGoogleLogo()
{
  Image googleLogo;
  googleLogo = browserInstance.Image(img =&amp;gt; img.Id == &amp;quot;logo&amp;quot;);
  Assert.IsTrue(googleLogo.Exists);
  Assert.AreEqual(&amp;quot;Google&amp;quot;, googleLogo.Alt);
}&lt;/pre&gt;
&lt;p&gt;
Next, check for the existence of the search criteria input box. WatiN refers to these
elements as Text Fields, using the TextField type. Additionally, this form field is
identified by its Name rather than its Id. In Google, the name given to the criteria
input is &amp;quot;q&amp;quot;.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public method named PageShouldContainSearchCriteriaInput and give it
the [TestMethod] attribute. 
&lt;/li&gt;
&lt;li&gt;
Search for and assert on the existence of a Text Field with the name &amp;quot;q&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void PageShouldContainSearchCriteriaInput()
{
  TextField criteriaInput;
  criteriaInput = browserInstance.TextField(tf =&amp;gt; tf.Name == &amp;quot;q&amp;quot;);
  Assert.IsTrue(criteriaInput.Exists);
}&lt;/pre&gt;
&lt;p&gt;
Finally, check for the existence of the search button using the Button method. In
our lambda expression, it is not important to know if the field is identified by a
Name property or an Id attribute, as WatiN supplies a IdOrName property to help us
find the element. The value to identify the button is &amp;quot;btnG&amp;quot;.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public method named PageShouldContainSearchButton and give it the [TestMethod]
attribute. 
&lt;/li&gt;
&lt;li&gt;
Search for and assert on the existence of a Button with the Id or Name of 'btnG&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Optionally, we can also check the value of the button, which is the text displayed
on the button on-screen. This text should be &amp;quot;Google Search&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void PageShouldContainSearchButton()
{
  Button searchButton;
  searchButton = browserInstance.Button(btn =&amp;gt; btn.IdOrName == &amp;quot;btnG&amp;quot;);
  Assert.IsTrue(searchButton.Exists);
  Assert.AreEqual(&amp;quot;Google Search&amp;quot;, searchButton.Value);
}&lt;/pre&gt;
&lt;h4&gt;Working with Style
&lt;/h4&gt;
&lt;p&gt;
WatiN can access properties on the DOM beyond just Text values and Alt attributes.
WatiN also has full access to the style that CSS has applied to an element. Let's
check out a few CSS properties, both those explicitly defined by WatiN and those implicitly
accessible through the WatiN framework.
&lt;/p&gt;
&lt;p&gt;
For our first style check, we'll take a look at the default font family used on the
Google Home Page. Font Family is one of the explicitly available style properties
on a WatiN element. Some others, like Color, Display, and Height are also explicitly
defined.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public test method named BodyShouldUseArialFontFamily. 
&lt;/li&gt;
&lt;li&gt;
Assert that the font family assigned to the body matches &amp;quot;arial, sans-serif&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void BodyShouldUseArialFontFamily()
{
  Assert.AreEqual(&amp;quot;arial, sans-serif&amp;quot;, browserInstance.Body.Style.FontFamily);
}&lt;/pre&gt;
&lt;p&gt;
For our second style check, we will look for an implicit style definition. At the
top of the Google Home Page is a series of links to other areas of Google, such as
Images, Videos, Maps, and News. At the end of this list is a More link, that when
clicked, displays a hidden DIV tag containing even more links, such as Books, Finance,
and Google Translate. Since we do not have any code in our test initialization that
interacts with the browser, and thus nothing that is clicking the More link, that
DIV should still have a hidden visibility. However, since Visibility isn't an explicitly
defined style property within WatiN, we need to use the GetAttributeValue method to
retrieve the current visibility setting.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public test method named MoreItemsShouldNotBeVisibleOnPageLoad. 
&lt;/li&gt;
&lt;li&gt;
Search for the More Items DIV. It's Id is &amp;quot;gbi&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Using the property lookup method, GetAttributeValue(string attributeName), check that
the Visibility is set to &amp;quot;hidden&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void MoreItemsShouldNotBeVisibleOnPageLoad()
{
  var googleBarMoreItems = browserInstance.Div(gbi =&amp;gt; gbi.Id == &amp;quot;gbi&amp;quot;);
  Assert.AreEqual(&amp;quot;hidden&amp;quot;, googleBarMoreItems.Style.GetAttributeValue(&amp;quot;visibility&amp;quot;));
}&lt;/pre&gt;
&lt;h3&gt;Exercise 2: Interacting with the Browser
&lt;/h3&gt;
&lt;p&gt;
Browser Integration tests are more than just loading a page and checking a few element
attributes. Our tests may also need to enter values into form fields, click links
and buttons, or interact with browser navigation like the back button. WatiN fully
supports all of these features in a very intuitive fashion.
&lt;/p&gt;
&lt;h4&gt;A new test class, this time with Search Capability
&lt;/h4&gt;
&lt;p&gt;
Create a new test class, similar to what we did in Exercise 1, calling the new test
class WhenViewingGoogleSearchResultsForComeJamWithUs. Also add in the TestInitialize
and TestCleanup methods that open and close the browser. However, this time, after
we load &lt;a href="http://www.google.com"&gt;http://www.google.com&lt;/a&gt;, enter a value into
the search criteria input and then click the Google Search button.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new class named WhenViewingGoogleSearchResultsForComeJamWithUs, similar to
what was done in Exercise 1. 
&lt;/li&gt;
&lt;li&gt;
Add in the TestInitialize and TestCleanup methods from Exercise 1. Name the Initialize
method WithAnInstanceOfTheBrowserSearchingGoogle. 
&lt;/li&gt;
&lt;li&gt;
After the code that initializes the IE class, find the search criteria Text Field
and set its value to &amp;quot;Come Jam With Us&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
After setting the Text Field value, click the Google Search button by calling the
Click() method on the Button class. 
&lt;/li&gt;
&lt;li&gt;
Compile. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;using Microsoft.VisualStudio.TestTools.UnitTesting;
using WatiN.Core;

namespace WatinSample
{
  [TestClass]
  public class WhenViewingGoogleSearchResultsForComeJamWithUs
  {
    Browser browserInstance;

    [TestInitialize]
    public void WithAnInstanceOfTheBrowserSearchingGoogle()
    {
      browserInstance = new IE(@&amp;quot;http://www.google.com&amp;quot;);
      TextField criteria =
        browserInstance.TextField(tf =&amp;gt; tf.Name == &amp;quot;q&amp;quot;);
      criteria.Value = &amp;quot;Come Jam With Us&amp;quot;;
      Button search =
        browserInstance.Button(btn =&amp;gt; btn.IdOrName == &amp;quot;btnG&amp;quot;);
      search.Click();
    }

    [TestCleanup]
    public void ShutdownBrowserWhenDone()
    {
      browserInstance.Close();
      browserInstance = null;
    }
  }
}&lt;/pre&gt;
&lt;p&gt;
With this code, or initialized test will load the Google Home Page and will conduct
a search for &amp;quot;Come Jam With Us&amp;quot;.
&lt;/p&gt;
&lt;h4&gt;Validating the Search Results Page
&lt;/h4&gt;
&lt;p&gt;
For our first verification, let's check the URL for the browser window. The search
result URL should contain the search criteria in the URL's query string; we can validate
this using the URL property on our instance of the Browser object.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public test method named BrowserUrlShouldContainSearchCriteria. 
&lt;/li&gt;
&lt;li&gt;
Validate that the current browser URL contains the search criteria information, &amp;quot;q=Come+Jam+With+Us&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void BrowserUrlShouldContainSearchCriteria()
{
  Assert.IsTrue(browserInstance.Url.Contains(@&amp;quot;q=Come+Jam+With+Us&amp;quot;));
}&lt;/pre&gt;
&lt;h4&gt;Finding Child Elements
&lt;/h4&gt;
&lt;p&gt;
With WatiN, we are not just limited to searching for items directly from the Browser
object. We can also search for child elements directly from their parent element or
any ancestor element. Our search results should contain a search result item linking
to the Come Jam With Us web site. The Google Results page contains a DIV identified
as &amp;quot;res&amp;quot; that serves as a container for all search result information. Rather
than checking that our Come Jam With Us link exists somewhere on the page, we should
search for it directly within the results DIV.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public test method named ResultsShouldContainLinkToComeJamWithUs. 
&lt;/li&gt;
&lt;li&gt;
From the browser instance, find a DIV identified as &amp;quot;res&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Assert that a link to &lt;a href="http://www.comejamwithus.org"&gt;http://www.comejamwithus.org&lt;/a&gt; exists
within the &amp;quot;res&amp;quot; DIV. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void ResultsShouldContainLinkToComeJamWithUs()
{
  Link comeJamWithUs;
  Div searchResults = browserInstance.Div(div =&amp;gt; div.IdOrName == &amp;quot;res&amp;quot;);
  comeJamWithUs =
    searchResults.Link(link =&amp;gt; link.Url == @&amp;quot;http://www.comejamwithus.org/&amp;quot;);
  Assert.IsTrue(comeJamWithUs.Exists);
}&lt;/pre&gt;
&lt;h4&gt;Inner Text verses InnerHtml
&lt;/h4&gt;
&lt;p&gt;
An element may contain many child elements. An anchor tag—&amp;lt;A href=&amp;quot;#&amp;quot;&amp;gt;—can
contain text, and child elements may make portions of that text bold, italic, underlined,
or even bright red. Through WatiN, we can access that inner content as straight text
without the formatting, or as the InnerHtml including all of the child elements.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create two public test methods, one named ResultsLinkContainsComeJamWithUsText and
the other named ResultsLinkContainsComeJamWithUsHtml. 
&lt;/li&gt;
&lt;li&gt;
In both methods, search for the results DIV, as we did in the previous test. 
&lt;/li&gt;
&lt;li&gt;
In both methods, search through the results DIV for a link with a URL matching &lt;a href="http://www.comejamwithus.org"&gt;http://www.comejamwithus.org&lt;/a&gt; 
&lt;/li&gt;
&lt;li&gt;
In the Text method, assert that the text of the link matches &amp;quot;Come Jam with us
(Software Development Study Group)&amp;quot;. Note that the value contains no child HTML
elements. 
&lt;/li&gt;
&lt;li&gt;
In the HTML method, assert that the InnerHtml of the link matches &amp;quot;&amp;lt;EM&amp;gt;Come
Jam with us&amp;lt;/EM&amp;gt; (Software Development Study Group)&amp;quot;. Note that for the
same link, we now have the emphasis tags surrounding Come Jam With Us. 
&lt;/li&gt;
&lt;li&gt;
Compile and run both tests. The tests should pass.&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void ResultsLinkContainsComeJamWithUsText()
{
  Link comeJamWithUs;
  Div searchResults = browserInstance.Div(div =&amp;gt; div.IdOrName == &amp;quot;res&amp;quot;);
  comeJamWithUs =
    searchResults.Link(link =&amp;gt; link.Url == @&amp;quot;http://www.comejamwithus.org/&amp;quot;);
  Assert.AreEqual(@&amp;quot;Come Jam with us (Software Development Study Group)&amp;quot;,
    comeJamWithUs.Text);
}

[TestMethod]
public void ResultsLinkContainsComeJamWithUsHtml()
{
  Link comeJamWithUs;
  Div searchResults = browserInstance.Div(div =&amp;gt; div.IdOrName == &amp;quot;res&amp;quot;);
  comeJamWithUs =
    searchResults.Link(link =&amp;gt; link.Url == @&amp;quot;http://www.comejamwithus.org/&amp;quot;);
  Assert.AreEqual(
    @&amp;quot;&amp;lt;EM&amp;gt;Come Jam with us&amp;lt;/EM&amp;gt; (Software Development Study Group)&amp;quot;,
    comeJamWithUs.InnerHtml);
}&lt;/pre&gt;
&lt;h4&gt;Back to the Start
&lt;/h4&gt;
&lt;p&gt;
As previously mentioned, we can also fully interact with the browser, itself. Our
test initialization started from the Google Home Page and performed a search. Using
functionality built in to WatiN, we can execute the browser's back navigation to return
to the previous page.
&lt;/p&gt;
&lt;p&gt;
For our next test, execute a back navigation and verify that the browser's URL matches &lt;a href="http://www.google.com/"&gt;http://www.google.com/&lt;/a&gt;.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a public test method named PageShouldHaveComeFromGoogleDotCom. 
&lt;/li&gt;
&lt;li&gt;
Execute back navigation in the browser by calling the Back() method on browserInstance. 
&lt;/li&gt;
&lt;li&gt;
Validate that the browser URL matches &lt;a href="http://www.google.com/"&gt;http://www.google.com/&lt;/a&gt;. 
&lt;/li&gt;
&lt;li&gt;
Compile and run the test. The test should pass.&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void PageShouldHaveComeFromGoogleDotCom()
{
  string previousUrl;
  browserInstance.Back();
  previousUrl = browserInstance.Url;
  Assert.AreEqual(@&amp;quot;http://www.google.com/&amp;quot;, previousUrl);
}&lt;/pre&gt;
&lt;h4&gt;Putting it all together
&lt;/h4&gt;
&lt;p&gt;
Some interactions on a page cause element properties to change. An example of this
is the More link from Exercise 1; when the end-user clicks the More link, the More
Items DIV appears because the link's click event changes the Visibility style property
of the DIV to visible. For our final test, we will use what we have learned to test
this functionality.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new public test method named MoreItemsShouldBeVisibleOnMoreLinkClick.&lt;/li&gt;
&lt;li&gt;
Search for the header bar of Google links, a DIV with an Id of &amp;quot;gbar&amp;quot;.&lt;/li&gt;
&lt;li&gt;
Within &amp;quot;gbar&amp;quot;, search for the More Items DIV by an Id or Name of &amp;quot;gbi&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Assert that the visibility style property has a value of &amp;quot;hidden&amp;quot;. 
&lt;/li&gt;
&lt;li&gt;
Within &amp;quot;gbar&amp;quot;, search for the More link by its class name, &amp;quot;gb3&amp;quot;.
Note that since a class attribute may contain multiple class definitions, this is
accomplished by validating that the class attribute contains the class you are searching
for. 
&lt;/li&gt;
&lt;li&gt;
Execute a Click event on the link. 
&lt;/li&gt;
&lt;li&gt;
Assert that the visibility style property of the More Items DIV has changed to &amp;quot;visible&amp;quot;. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;[TestMethod]
public void MoreItemsShouldBeVisibleOnMoreLinkClick()
{
  var googleBar = browserInstance.Div(gbar =&amp;gt; gbar.Id == &amp;quot;gbar&amp;quot;);
  var googleBarMoreItems = googleBar.Div(gbi =&amp;gt; gbi.Id == &amp;quot;gbi&amp;quot;);
  Assert.AreEqual(&amp;quot;hidden&amp;quot;,
    googleBarMoreItems.Style.GetAttributeValue(&amp;quot;visibility&amp;quot;));
  var googleBarMoreLink =
    googleBar.Link(link =&amp;gt; link.ClassName.Contains(&amp;quot;gb3&amp;quot;));
  googleBarMoreLink.Click();
  Assert.AreEqual(&amp;quot;visible&amp;quot;,
    googleBarMoreItems.Style.GetAttributeValue(&amp;quot;visibility&amp;quot;));
}&lt;/pre&gt;
&lt;h4&gt;That's It
&lt;/h4&gt;
&lt;p&gt;
Now that we have spent some time on basic properties, interactions, and style sheets
within the WatiN framework, hopefully you can apply this to your own application and
get started with your own browser-based integration tests. If you would like more
information, I encourage you to check out the WatiN site at &lt;a href="http://watin.sourceforge.net"&gt;http://watin.sourceforge.net&lt;/a&gt;.
And as always, if you have any questions, drop me a line.
&lt;/p&gt;
&lt;div style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; display: inline; float: none; padding-top: 0px" id="scid:0767317B-992E-4b12-91E0-4F059A8CECA8:9a272a14-7cc1-4234-a4bc-8531ce7b8597" class="wlWriterEditableSmartContent"&gt;Technorati
Tags: &lt;a href="http://technorati.com/tags/WatiN" rel="tag"&gt;WatiN&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Jam+Session" rel="tag"&gt;Jam
Session&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Coding+Exercise" rel="tag"&gt;Coding
Exercise&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Learn+to+Code" rel="tag"&gt;Learn to
Code&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Browser+Testing" rel="tag"&gt;Browser Testing&lt;/a&gt;
&lt;/div&gt;
&lt;img width="0" height="0" src="http://www.cptloadtest.com/aggbug.ashx?id=5a613524-5607-4fbb-a89f-74b5afb7c5e9" /&gt;</description>
      <comments>http://www.cptloadtest.com/CommentView,guid,5a613524-5607-4fbb-a89f-74b5afb7c5e9.aspx</comments>
      <category>Learn to Code</category>
      <category>Testing</category>
      <category>Tools</category>
    </item>
    <item>
      <trackback:ping>http://www.cptloadtest.com/Trackback.aspx?guid=1d5d9618-a51a-471f-a5ec-7381c916f11d</trackback:ping>
      <pingback:server>http://www.cptloadtest.com/pingback.aspx</pingback:server>
      <pingback:target>http://www.cptloadtest.com/PermaLink,guid,1d5d9618-a51a-471f-a5ec-7381c916f11d.aspx</pingback:target>
      <dc:creator>Jay Harris</dc:creator>
      <wfw:comment>http://www.cptloadtest.com/CommentView,guid,1d5d9618-a51a-471f-a5ec-7381c916f11d.aspx</wfw:comment>
      <wfw:commentRss>http://www.cptloadtest.com/SyndicationService.asmx/GetEntryCommentsRss?guid=1d5d9618-a51a-471f-a5ec-7381c916f11d</wfw:commentRss>
      <slash:comments>3</slash:comments>
      <body xmlns="http://www.w3.org/1999/xhtml">
        <p>
On Tuesday, February 9th, I was scheduled lead a jam session for Come Jam With Us,
the software developer study group in Ann Arbor. The session was to be on ASP.NET
MVC 2, aimed to give attendees enough of an introduction to the product to empower
developers to be able to start coding their own ASP.NET MVC 2 projects. Unfortunately,
Mother Nature did not cooperate that day, half of the state of Michigan seemingly
shut down under 8" of snow, and the session was cancelled and rescheduled for
February 23rd. The goal of these Learn to Code exercises is to give you an introduction
to building applications with ASP.NET MVC 2. In the near future, I also hope to provide
a screen cast of these same exercises.
</p>
        <h3>About this Exercise
</h3>
        <p>
This coding exercise is designed to give you an introduction to ASP.NET MVC 2. In
this exercise, developers will create their first database-driven ASP.NET MVC 2 application
within Visual Studio, primarily using code generation built in to Visual Studio. Developers
performing this exercise should be familiar with ASP.NET development and Visual Studio,
but no previous experience with ASP.NET MVC is required.
</p>
        <h3>Prerequisites
</h3>
        <p>
You will need few things for ASP.NET MVC 2 application development and to complete
these exercises. Please complete the following prerequisites prior to moving on. The
session is designed to be completed in about an hour, but prerequisite setup is not
included in that time.
</p>
        <ul>
          <li>
Install <a href="http://www.microsoft.com/visualstudio/">Microsoft Visual Studio 2008
SP1</a> or <a href="http://www.microsoft.com/express/Web/">Visual Web Developer 2008
Express</a> with SP1 
</li>
          <li>
Install <a href="http://www.microsoft.com/sqlserver/2008/">Microsoft SQL Server</a> or <a href="http://www.microsoft.com/express/Database/">SQL
Server Express</a></li>
          <li>
Download and install the latest <a title="ASP.NET MVC 2 RC 2 on Microsoft" href="http://www.microsoft.com/downloads/details.aspx?FamilyID=7aba081a-19b9-44c4-a247-3882c8f749e3&amp;displaylang=en">ASP.NET
MVC 2 RC 2 bits from Microsoft</a></li>
        </ul>
        <h3>Exercise 0: Getting Started
</h3>
        <h4>Creating a Project
</h4>
        <p>
Before any coding can occur, the first thing that we have to do is create a new ASP.NET
MVC 2 project from within Visual Studio.
</p>
        <ol>
          <li>
In Visual Studio, create a new "ASP.NET MVC 2 Web Application" named MvcJamSession.
You can create your project in either Visual Basic or Visual C#, though all of the
examples in this post will be in C#. 
</li>
          <li>
After selecting the project type and solution/project name, you will be prompted for
creating a unit test project. Select "Yes." Though we will not be using
these tests in this exercise, we will be in future installments. 
</li>
          <li>
Be sure that your MvcJamSession.Test project includes a project reference back to
your MvcJamSession project. 
</li>
          <li>
Compile and run. Your browser should display a blue web site showing "Welcome
to ASP.NET MVC!" Congratulations. You now have your first ASP.NET MVC application. 
</li>
        </ol>
        <h4>Convention-Based Development
</h4>
        <p>
          <a href="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/SolutionExplorer-MvcJamSession.jpg">
            <img style="border-right-width: 0px; margin: 0px 0px 5px 15px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="Project Folder Structure of MVC Jam Session Project" border="0" alt="Project Folder Structure of MVC Jam Session Project" align="right" src="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/SolutionExplorer-MvcJamSession_thumb.jpg" width="200" height="382" />
          </a> Development
within ASP.NET MVC is based on convention over configuration. Certain naming conventions
are built in the system to eliminate the amount of boiler-plate code that you need
to recreate. That is not to say that you <em>must</em> follow these naming conventions—you
may use whatever convention you like—but by straying from the standardized conventions,
you will be creating a lot of extra work for yourself. With our new ASP.NET MVC 2
Project, a few of these conventions are immediately visible.
</p>
        <ul>
          <li>
            <em>Controllers</em> folder. This is where all Controller classes must go. Individual
classes must be named <em>&lt;Name&gt;Controller</em>. The default project includes <em>HomeController</em>,
which governs the Home and About actions, and <em>AccountController</em>, which governs
the log in, log out, change password, and registration actions. If we were to make
an application that manages Widgets, we would likely have a class named <em>WidgetController</em>. 
</li>
          <li>
            <em>Views</em> folder. This is where all of the Views must go. By default, views are
paired one-to-one with controller actions, such as one view for new user registration
and another view for changing your password. Views are also separated into folders
matching the associated controller name—<em>/Views/&lt;ControllerName&gt;/&lt;ActionName&gt;</em>.
Thus, <em>HomeController</em>'s About action is associated with the <em>/Views/Home/About</em> view. 
<br />
The Views folder also contains a <em>Shared</em> folder. This folder is where any
common views, such as a Master Page, would reside. The <em>Shared</em> folder is also
where the ViewEngine cascades to when it can't find an appropriate view in the /<em>Views/&lt;ControllerName&gt;</em> folder;
if <em>/Views/Home/About</em> didn't exist, the ViewEngine would look for <em>/Views/Shared/About</em>.
This can come in handy for common pages shared by all controllers, such as an error
page. 
</li>
        </ul>
        <h3>Session Exercise 1: Building an Application
</h3>
        <p>
Using the project we just created, we're going to create an application that manage
a list of employees, including their name, job title, date of hire, and date of termination.
Though the default project is a great help on some applications, it can get in the
way on others; we're not going to need any account services in our application, so
we need to first trim the project down a little.
</p>
        <ol>
          <li>
Delete the entire <em>Account</em> folder under /Views/. 
</li>
          <li>
Delete <em>AccountController.cs</em> from the Controllers folder. 
</li>
          <li>
Delete shared partial view <em>/Views/Shared/LogOnUserControl.ascx</em>. 
</li>
          <li>
Delete reference to this partial view by removing the "LoginDisplay" DIV
from<em> /Views/Shared/Site.Master</em>, lines 18-20. 
</li>
          <li>
Removing <em>LoginDisplay</em> will cause a slight layout problem from the CSS. To
fix it, modify the <em>#menucontainer</em> definition in <em>/Content/Site.css</em> on
line 263. 
<br /><pre class="css:nocontrols" name="code">#menucontainer
{
    padding-top:40px;
}</pre></li>
          <li>
Save all, compile, and Run. The site should be functioning normally, but without the
Log In link in the top right of the home page. 
</li>
        </ol>
        <h4>Creating a Database
</h4>
        <p>
Like any dynamic web site, we need a storage mechanism. Create a database in SQL Server
or SQL Server Express with an Employee table containing columns for name, job title,
hired date, and termination date, as well as an identity column for finding records.<a href="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/MvcJamSession-EmployeeTable_1.jpg"><img style="border-right-width: 0px; margin: 5px 0px 5px 15px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="MvcJamSession-EmployeeTable" border="0" alt="MvcJamSession-EmployeeTable" align="right" src="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/MvcJamSession-EmployeeTable_thumb_1.jpg" width="250" height="283" /></a></p>
        <ul>
          <li>
For those with SQL Server Express, create a new database through right-clicking the
App_Data folder, and adding a new item. The new item should be a SQL Server Database. 
</li>
          <li>
For those with SQL Server, create a new database through SQL Server Management Studio
(SSMS). 
</li>
        </ul>
        <ol>
          <li>
Create a new table called <em>Employee</em>. 
</li>
          <li>
Create the following columns within the new table: 
<ul><li>
Id (primary key, identity, int, not null) 
</li><li>
Name (varchar(50), not null) 
</li><li>
JobTitle (varchar(50), not null) 
</li><li>
HiredOn (date, not null) 
</li><li>
TerminatedOn (date, null) 
</li></ul></li>
          <li>
Add a record or two to the table for good measure. 
</li>
        </ol>
        <h4>Creating a Model
</h4>
        <p>
The next thing we need to create is our Model. Not only is it the code representation
for our business entity (An Employee class contains Name, JobTitle, HiredOn, and TerminatedOn
properties), it is also responsible for <em>how</em> to get, save, or delete data
from the database. We will be using Microsoft Entity Framework through Visual Studio
to generate our model for us.
</p>
        <ol>
          <li>
To create our new Model, right-click the <em>Model</em> folder and select Add New
Item. The new item should be an ADO.NET Entity Data Model, found within the Data category
of the Add New Item dialog. Name it MvcJamSessionEntities. 
</li>
          <li>
Generate the Model from a database, connecting to your MVC Jam Session database created
in the previous section. On the step where you select your database connection, be
sure to allow Visual Studio to add the connection string to your web.config by checking
the appropriate checkbox. 
</li>
          <li>
The <em>Employee</em> table is the only Data Object that needs to be generated. 
</li>
          <li>
When the dialog completes, your Entity Data Model (a .edmx file) should be displayed
in design view. 
</li>
          <li>
Save all and compile. 
</li>
        </ol>
        <h4>Creating a Controller
</h4>
        <p>
Now that our model is in place, we need to create a controller. The controller is
responsible for managing all interaction between the end-user and the application,
including identifying what data to get, save, or delete. (Remember, though the Controller
is responsible for <em>what</em>, the Model is responsible for <em>how</em>.)
</p>
        <ol>
          <li>
To create our new Controller, right-click the <em>Controller</em> folder and select
Add Controller. Since the name matters in ASP.NET MVC's convention-over-configuration
style, name it EmployeeController.cs (or .vb, if you happen to be working in Visual
Basic .NET). Be sure to check the "Add action methods for Create, Update, Detail,
and Delete scenarios" as we will be using them later. 
</li>
          <li>
We now have a basic controller, but it doesn't do anything yet. First, we want to
modify our Index action, as it is the default action in the controller. We will use
this action to list all of the Employees that currently exist in our database. Your
Index() action method should contain the following code: 
<br /><pre class="csharp:nocontrols" name="code">var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
return View(_entities.Employee.ToList());</pre></li>
          <li>
Save all and compile. 
</li>
        </ol>
        <h4>Creating a View
</h4>
        <p>
We have our Model and we have our Controller, so it is time for our View. The View
is responsible for display only—it should contain virtually no logic. Our controller
doesn't do anything yet, but we can at least get the base file structure and navigation
in place. Since our Model governs <em>how</em> and our Controller governs <em>what</em>,
think of the View as governing <em>where</em>, as it is responsible for deciding where
each data element gets displayed on your page.
</p>
        <ol>
          <li>
Once you have saved and compiled your Controller, right-click on the Index action
name and select Add View. 
</li>
          <li>
The View should be named Index, just like your Controller action. Also, make your
View strongly typed to the Employee model. Since a list of Employees is being passed
to the View from the Controller, making the View strongly-typed prevents us from having
to cast our View Model from object to Employee. Finally, since we are providing a
list of Employees, the View Content should be a List. 
</li>
          <li>
The <em>Employee</em> folder should be automatically created under the <em>/Views/</em> folder,
and the <em>Index.aspx</em> View inside of this new <em>Employee</em> folder. 
</li>
          <li>
The last thing we need to do is provide some navigation to this view. Open up <em>/Views/Shared/Site.Master</em> and
add an ActionLink within the Menu Container section to the Index action of the Employee
controller. When you are done, the Menu Container should look like this: 
<br /><pre class="csharp:nocontrols" name="code">&lt;ul id="menu"&gt;              
  &lt;li&gt;&lt;%= Html.ActionLink("Home", "Index", "Home")%&gt;&lt;/li&gt;
  &lt;li&gt;&lt;%= Html.ActionLink("About", "About", "Home")%&gt;&lt;/li&gt;
  &lt;li&gt;&lt;%= Html.ActionLink("Employees", "Index", "Employee")%&gt;&lt;/li&gt;
&lt;/ul&gt;</pre></li>
          <li>
Save all and run. When you navigate to your Employees link, you should get a list
of all employees currently in the database with an Edit, Details, and Delete links. 
</li>
        </ol>
        <blockquote>
          <h5>New in ASP.NET MVC 2: Strongly Typed HTML Helpers
</h5>
          <p>
In the previous version of ASP.NET MVC, HTML helpers were simple generic classes.
Generated views were full of Magic Strings for each property in your model, such as
&lt;%= Html.TextBox("Name") %&gt;, opening the door for a fat-fingered property
name. MVC 2 includes strongly-typed HTML helpers on strongly-typed views. For form-based
views, use the new strongly-typed "For" methods, such as &lt;%= Html.TextBoxFor(model
=&gt; model.Name) %&gt; or &lt;%= Html.EditorFor(model =&gt; model.Name) %&gt; to
eliminate the risk of incorrectly entering a property name. For display fields, use
Html.DisplayFor() to provide similar benefits for your read-only data, including the
elimination of HTML encoding for each field.
</p>
        </blockquote>
        <h4>Adding New Data
</h4>
        <p>
A list of employees is great, but we also need the ability to manipulate that data.
First, let's start with the ability to create new data.
</p>
        <ol>
          <li>
Within Visual Studio, open <em>/Controllers/EmployeeController</em> and navigate to
the POST Create action. POST is an HTTP verb associated with pushing data to the server
in the HTTP header, commonly associated with form submits. GET is the HTTP verb for
pure retrieval, commonly associated with a direct URL request, such as clicking a
link. In this case, the POST action can be identified by the [HttpPost] decoration. 
</li>
          <li>
The method arguments currently include only a FormCollection object that will contain
all of the header values associated with the POST. However, MVC is smart; it can automatically
transform this collection into a type-safe Model, based on the names of the header
variables (the identities of the HTML form inputs match the names of the Model's properties).
The one exception is the Id attribute, which is available in the Model but not populated
until after the object is saved to the database. To get strong typing, and avoid having
to manually cast or map form collection data to a new instance of our Model, change
the method signature to the following: 
<br /><pre class="csharp:nocontrols" name="code">public ActionResult Create([Bind(Exclude="Id")] Employee newEmployee)</pre></li>
          <li>
Now that we have a populated Model, we just need to save the Model to the database
and redirect back to the List when we are done. Do this by replacing the contents
of the action method with the following code: 
<br /><pre class="csharp:nocontrols" name="code">try
{
  if (!ModelState.IsValid) return View();

  var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
  _entities.AddToEmployee(employee);
  _entities.SaveChanges();
  return RedirectToAction("Index");
}
catch
{
  return View();
}</pre></li>
          <li>
Save and compile. 
</li>
          <li>
Now we must create the View for adding data. As we did with Index, we can create the
view by right-clicking the Action method and selecting Add View. The view content
should be Create. 
</li>
          <li>
The only modification for the Create view is the Id property. The generated code creates
an input for this property, but it is not needed since the column value is auto-populated
by the database when saving the entity. Remove this input and label from the form. 
</li>
          <li>
Save and run. You should now be able to add new records to the database. 
</li>
        </ol>
        <blockquote>
          <h5>New in ASP.NET MVC2: Better Verb Attributes
</h5>
          <p>
In the first version of ASP.NET MVC, HTTP Verb conditions were placed on an Action
via the AcceptVerbsAttribute, such as the Create action's [AcceptVerbs(HttpVerbs.Post)].
In ASP.NET MVC 2, these attributes have been simplified with the introduction of the
HttpGetAttribute, HttpDeleteAttribute, HttpPostAttribute, and HttpPutAttribute.
</p>
        </blockquote>
        <h4>Routing and Updates
</h4>
        <p>
The end user can view a list of Employees, and can create new employees, but when
the end user clicks the Edit or Detail links, they get an error since these Views
haven't been created yet and the Actions are not implemented. One by one, we will
get the new views in place.
</p>
        <ol>
          <li>
Within Visual Studio, open <em>/Controllers/EmployeeController</em> and navigate to
the Details action. You may notice that the method already accepts an integer as input,
and shows example usage of the action in a commented URL above the method: <em>/Employee/Details/5</em>.
This integer is the identity value of the Employee record, and is already populated
in the links of our List view created in the previous section. 
</li>
          <li>
Within Visual Studio, open <em>Global.asax</em> and navigate to the RegisterRoutes
method. The default route for ASP.NET MVC is <em>/{controller}/{action}/{id}/</em>.
By parsing out any URL into the application, MVC can determine which Controller to
use, which Action to execute, and which arguments to pass to the Action. Later portions
of the URL path are often optional, and when not specified, are replaced with the
default values: Home, Index, and String.Empty. The URLs of "/", "/Home",
"/Home/", "/Home/Index", and "/Home/Index/" are all
equivalent URLs in the eyes of ASP.NET MVC. 
</li>
        </ol>
        <p>
Go back to the Employee controller. Now that we know what the integer argument is
for, we need to retrieve the Employee matching the associated identity and pass it
to a view for editing or display.
</p>
        <ol>
          <li>
Replace the contents of the GET version of the Edit action method with the following
code to retrieve the Employee from the database that matches the identity specified
in the route: 
<br /><pre class="csharp:nocontrols" name="code">var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
return View(_entities.Employee.Where(emp =&gt; emp.Id == id).First());</pre></li>
          <li>
Save and compile. 
</li>
          <li>
Right-click the Action and add the View. The View should still be strongly typed,
but this time the view content should be Edit. 
</li>
          <li>
The Details and GET Delete actions are largely similar as the GET Edit action, except
that the view is labels instead of text boxes. Repeat the above three steps for the
Details action method with a Details view content and for the GET Delete action method
with a Delete view content. 
</li>
          <li>
As with the Create view, the Id property should be removed from the Edit view, as
it is not an item that should be edited by the end user. 
</li>
          <li>
Save and run. You should see the details of an Employee when clicking on the Edit
and Details links. 
</li>
          <li>
We can view Employee details within the Edit form, but when we make changes and submit,
nothing happens. We need to modify the POST Edit action to save our changes back to
the database. The default POST Edit action accepts an id and a FormCollection as input
arguments, but similarly to the POST Create action, we can change this to use our
strongly typed model to avoid having to cast or map data. However, unlike our Create
action, we need to bind the id property so that the system knows which record to update.
To make these modifications, replace the POST Edit signature with the following: 
<br /><pre class="csharp:nocontrols" name="code">public ActionResult Edit(MvcJamSession.Models.Employee employee)</pre></li>
          <li>
Replace the contents of the POST Edit action method with the following code to save
the changes to the database: 
<br /><pre class="csharp:nocontrols" name="code">try
{
  if (!ModelState.IsValid) return View();

  var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
  var _originalEmployee =
    _entities.Employee.Where(emp =&gt; emp.Id == employee.Id).First();
  _entities.ApplyPropertyChanges(_originalEmployee.EntityKey.EntitySetName,
                                 employee);
  _entities.SaveChanges();
  return RedirectToAction("Index");
}
catch
{
  return View();
}</pre></li>
          <li>
Replace the signature of the POST Delete action method with the following code to
provide strong typing on our model: 
<br /><pre class="csharp:nocontrols" name="code">public ActionResult Delete(MvcJamSession.Models.Employee employee)</pre></li>
          <li>
Finally, replace the contents of the POST Delete action method with the following
code to delete a record: 
<br /><pre class="csharp:nocontrols" name="code">var _entites = new MvcJamSession.Models.MvcJamSessionEntities();
var originalEmployee =
  _entites.Employee.Where(emp =&gt; emp.Id == deletedEmployee.Id).First();
_entites.DeleteObject(originalEmployee);
_entites.SaveChanges();
return RedirectToAction("Index");</pre></li>
          <li>
Save, compile, and run. You should now be able to modify existing Employee records. 
</li>
        </ol>
        <p>
We now have a fully-functional ASP.NET MVC application to manage our Employee records.
Congratulations!
</p>
        <div style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; display: inline; float: none; padding-top: 0px" id="scid:0767317B-992E-4b12-91E0-4F059A8CECA8:13d5b303-4a7f-4ac9-93a3-f208ffe24fe0" class="wlWriterEditableSmartContent">Technorati
Tags: <a href="http://technorati.com/tags/ASP.NET+MVC" rel="tag">ASP.NET MVC</a>,<a href="http://technorati.com/tags/MVC" rel="tag">MVC</a>,<a href="http://technorati.com/tags/Jam+Session" rel="tag">Jam
Session</a>,<a href="http://technorati.com/tags/Coding+Exercise" rel="tag">Coding
Exercise</a>,<a href="http://technorati.com/tags/Learn+to+Code" rel="tag">Learn to
Code</a></div>
        <img width="0" height="0" src="http://www.cptloadtest.com/aggbug.ashx?id=1d5d9618-a51a-471f-a5ec-7381c916f11d" />
      </body>
      <title>Learn to Code ASP.NET MVC 2 : Introduction to ASP.NET MVC 2</title>
      <guid isPermaLink="false">http://www.cptloadtest.com/PermaLink,guid,1d5d9618-a51a-471f-a5ec-7381c916f11d.aspx</guid>
      <link>http://www.cptloadtest.com/2010/02/23/Learn-To-Code-ASPNET-MVC-2-Introduction-To-ASPNET-MVC-2.aspx</link>
      <pubDate>Tue, 23 Feb 2010 22:38:00 GMT</pubDate>
      <description>&lt;p&gt;
On Tuesday, February 9th, I was scheduled lead a jam session for Come Jam With Us,
the software developer study group in Ann Arbor. The session was to be on ASP.NET
MVC 2, aimed to give attendees enough of an introduction to the product to empower
developers to be able to start coding their own ASP.NET MVC 2 projects. Unfortunately,
Mother Nature did not cooperate that day, half of the state of Michigan seemingly
shut down under 8&amp;quot; of snow, and the session was cancelled and rescheduled for
February 23rd. The goal of these Learn to Code exercises is to give you an introduction
to building applications with ASP.NET MVC 2. In the near future, I also hope to provide
a screen cast of these same exercises.
&lt;/p&gt;
&lt;h3&gt;About this Exercise
&lt;/h3&gt;
&lt;p&gt;
This coding exercise is designed to give you an introduction to ASP.NET MVC 2. In
this exercise, developers will create their first database-driven ASP.NET MVC 2 application
within Visual Studio, primarily using code generation built in to Visual Studio. Developers
performing this exercise should be familiar with ASP.NET development and Visual Studio,
but no previous experience with ASP.NET MVC is required.
&lt;/p&gt;
&lt;h3&gt;Prerequisites
&lt;/h3&gt;
&lt;p&gt;
You will need few things for ASP.NET MVC 2 application development and to complete
these exercises. Please complete the following prerequisites prior to moving on. The
session is designed to be completed in about an hour, but prerequisite setup is not
included in that time.
&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;
Install &lt;a href="http://www.microsoft.com/visualstudio/"&gt;Microsoft Visual Studio 2008
SP1&lt;/a&gt; or &lt;a href="http://www.microsoft.com/express/Web/"&gt;Visual Web Developer 2008
Express&lt;/a&gt; with SP1 
&lt;/li&gt;
&lt;li&gt;
Install &lt;a href="http://www.microsoft.com/sqlserver/2008/"&gt;Microsoft SQL Server&lt;/a&gt; or &lt;a href="http://www.microsoft.com/express/Database/"&gt;SQL
Server Express&lt;/a&gt; 
&lt;/li&gt;
&lt;li&gt;
Download and install the latest &lt;a title="ASP.NET MVC 2 RC 2 on Microsoft" href="http://www.microsoft.com/downloads/details.aspx?FamilyID=7aba081a-19b9-44c4-a247-3882c8f749e3&amp;amp;displaylang=en"&gt;ASP.NET
MVC 2 RC 2 bits from Microsoft&lt;/a&gt; 
&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Exercise 0: Getting Started
&lt;/h3&gt;
&lt;h4&gt;Creating a Project
&lt;/h4&gt;
&lt;p&gt;
Before any coding can occur, the first thing that we have to do is create a new ASP.NET
MVC 2 project from within Visual Studio.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
In Visual Studio, create a new &amp;quot;ASP.NET MVC 2 Web Application&amp;quot; named MvcJamSession.
You can create your project in either Visual Basic or Visual C#, though all of the
examples in this post will be in C#. 
&lt;/li&gt;
&lt;li&gt;
After selecting the project type and solution/project name, you will be prompted for
creating a unit test project. Select &amp;quot;Yes.&amp;quot; Though we will not be using
these tests in this exercise, we will be in future installments. 
&lt;/li&gt;
&lt;li&gt;
Be sure that your MvcJamSession.Test project includes a project reference back to
your MvcJamSession project. 
&lt;/li&gt;
&lt;li&gt;
Compile and run. Your browser should display a blue web site showing &amp;quot;Welcome
to ASP.NET MVC!&amp;quot; Congratulations. You now have your first ASP.NET MVC application. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;h4&gt;Convention-Based Development
&lt;/h4&gt;
&lt;p&gt;
&lt;a href="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/SolutionExplorer-MvcJamSession.jpg"&gt;&lt;img style="border-right-width: 0px; margin: 0px 0px 5px 15px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="Project Folder Structure of MVC Jam Session Project" border="0" alt="Project Folder Structure of MVC Jam Session Project" align="right" src="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/SolutionExplorer-MvcJamSession_thumb.jpg" width="200" height="382" /&gt;&lt;/a&gt; Development
within ASP.NET MVC is based on convention over configuration. Certain naming conventions
are built in the system to eliminate the amount of boiler-plate code that you need
to recreate. That is not to say that you &lt;em&gt;must&lt;/em&gt; follow these naming conventions—you
may use whatever convention you like—but by straying from the standardized conventions,
you will be creating a lot of extra work for yourself. With our new ASP.NET MVC 2
Project, a few of these conventions are immediately visible.
&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;
&lt;em&gt;Controllers&lt;/em&gt; folder. This is where all Controller classes must go. Individual
classes must be named &lt;em&gt;&amp;lt;Name&amp;gt;Controller&lt;/em&gt;. The default project includes &lt;em&gt;HomeController&lt;/em&gt;,
which governs the Home and About actions, and &lt;em&gt;AccountController&lt;/em&gt;, which governs
the log in, log out, change password, and registration actions. If we were to make
an application that manages Widgets, we would likely have a class named &lt;em&gt;WidgetController&lt;/em&gt;. 
&lt;/li&gt;
&lt;li&gt;
&lt;em&gt;Views&lt;/em&gt; folder. This is where all of the Views must go. By default, views are
paired one-to-one with controller actions, such as one view for new user registration
and another view for changing your password. Views are also separated into folders
matching the associated controller name—&lt;em&gt;/Views/&amp;lt;ControllerName&amp;gt;/&amp;lt;ActionName&amp;gt;&lt;/em&gt;.
Thus, &lt;em&gt;HomeController&lt;/em&gt;'s About action is associated with the &lt;em&gt;/Views/Home/About&lt;/em&gt; view. 
&lt;br /&gt;
The Views folder also contains a &lt;em&gt;Shared&lt;/em&gt; folder. This folder is where any
common views, such as a Master Page, would reside. The &lt;em&gt;Shared&lt;/em&gt; folder is also
where the ViewEngine cascades to when it can't find an appropriate view in the /&lt;em&gt;Views/&amp;lt;ControllerName&amp;gt;&lt;/em&gt; folder;
if &lt;em&gt;/Views/Home/About&lt;/em&gt; didn't exist, the ViewEngine would look for &lt;em&gt;/Views/Shared/About&lt;/em&gt;.
This can come in handy for common pages shared by all controllers, such as an error
page. 
&lt;/li&gt;
&lt;/ul&gt;
&lt;h3&gt;Session Exercise 1: Building an Application
&lt;/h3&gt;
&lt;p&gt;
Using the project we just created, we're going to create an application that manage
a list of employees, including their name, job title, date of hire, and date of termination.
Though the default project is a great help on some applications, it can get in the
way on others; we're not going to need any account services in our application, so
we need to first trim the project down a little.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Delete the entire &lt;em&gt;Account&lt;/em&gt; folder under /Views/. 
&lt;/li&gt;
&lt;li&gt;
Delete &lt;em&gt;AccountController.cs&lt;/em&gt; from the Controllers folder. 
&lt;/li&gt;
&lt;li&gt;
Delete shared partial view &lt;em&gt;/Views/Shared/LogOnUserControl.ascx&lt;/em&gt;. 
&lt;/li&gt;
&lt;li&gt;
Delete reference to this partial view by removing the &amp;quot;LoginDisplay&amp;quot; DIV
from&lt;em&gt; /Views/Shared/Site.Master&lt;/em&gt;, lines 18-20. 
&lt;/li&gt;
&lt;li&gt;
Removing &lt;em&gt;LoginDisplay&lt;/em&gt; will cause a slight layout problem from the CSS. To
fix it, modify the &lt;em&gt;#menucontainer&lt;/em&gt; definition in &lt;em&gt;/Content/Site.css&lt;/em&gt; on
line 263. 
&lt;br /&gt;
&lt;pre class="css:nocontrols" name="code"&gt;#menucontainer
{
    padding-top:40px;
}&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Save all, compile, and Run. The site should be functioning normally, but without the
Log In link in the top right of the home page. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;h4&gt;Creating a Database
&lt;/h4&gt;
&lt;p&gt;
Like any dynamic web site, we need a storage mechanism. Create a database in SQL Server
or SQL Server Express with an Employee table containing columns for name, job title,
hired date, and termination date, as well as an identity column for finding records.&lt;a href="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/MvcJamSession-EmployeeTable_1.jpg"&gt;&lt;img style="border-right-width: 0px; margin: 5px 0px 5px 15px; display: inline; border-top-width: 0px; border-bottom-width: 0px; border-left-width: 0px" title="MvcJamSession-EmployeeTable" border="0" alt="MvcJamSession-EmployeeTable" align="right" src="http://www.cptloadtest.com/content/binary/WindowsLiveWriter/ASP.NETMVCJamSession_14C62/MvcJamSession-EmployeeTable_thumb_1.jpg" width="250" height="283" /&gt;&lt;/a&gt; 
&lt;/p&gt;
&lt;ul&gt;
&lt;li&gt;
For those with SQL Server Express, create a new database through right-clicking the
App_Data folder, and adding a new item. The new item should be a SQL Server Database. 
&lt;/li&gt;
&lt;li&gt;
For those with SQL Server, create a new database through SQL Server Management Studio
(SSMS). 
&lt;/li&gt;
&lt;/ul&gt;
&lt;ol&gt;
&lt;li&gt;
Create a new table called &lt;em&gt;Employee&lt;/em&gt;. 
&lt;/li&gt;
&lt;li&gt;
Create the following columns within the new table: 
&lt;ul&gt;
&lt;li&gt;
Id (primary key, identity, int, not null) 
&lt;/li&gt;
&lt;li&gt;
Name (varchar(50), not null) 
&lt;/li&gt;
&lt;li&gt;
JobTitle (varchar(50), not null) 
&lt;/li&gt;
&lt;li&gt;
HiredOn (date, not null) 
&lt;/li&gt;
&lt;li&gt;
TerminatedOn (date, null) 
&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;li&gt;
Add a record or two to the table for good measure. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;h4&gt;Creating a Model
&lt;/h4&gt;
&lt;p&gt;
The next thing we need to create is our Model. Not only is it the code representation
for our business entity (An Employee class contains Name, JobTitle, HiredOn, and TerminatedOn
properties), it is also responsible for &lt;em&gt;how&lt;/em&gt; to get, save, or delete data
from the database. We will be using Microsoft Entity Framework through Visual Studio
to generate our model for us.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
To create our new Model, right-click the &lt;em&gt;Model&lt;/em&gt; folder and select Add New
Item. The new item should be an ADO.NET Entity Data Model, found within the Data category
of the Add New Item dialog. Name it MvcJamSessionEntities. 
&lt;/li&gt;
&lt;li&gt;
Generate the Model from a database, connecting to your MVC Jam Session database created
in the previous section. On the step where you select your database connection, be
sure to allow Visual Studio to add the connection string to your web.config by checking
the appropriate checkbox. 
&lt;/li&gt;
&lt;li&gt;
The &lt;em&gt;Employee&lt;/em&gt; table is the only Data Object that needs to be generated. 
&lt;/li&gt;
&lt;li&gt;
When the dialog completes, your Entity Data Model (a .edmx file) should be displayed
in design view. 
&lt;/li&gt;
&lt;li&gt;
Save all and compile. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;h4&gt;Creating a Controller
&lt;/h4&gt;
&lt;p&gt;
Now that our model is in place, we need to create a controller. The controller is
responsible for managing all interaction between the end-user and the application,
including identifying what data to get, save, or delete. (Remember, though the Controller
is responsible for &lt;em&gt;what&lt;/em&gt;, the Model is responsible for &lt;em&gt;how&lt;/em&gt;.)
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
To create our new Controller, right-click the &lt;em&gt;Controller&lt;/em&gt; folder and select
Add Controller. Since the name matters in ASP.NET MVC's convention-over-configuration
style, name it EmployeeController.cs (or .vb, if you happen to be working in Visual
Basic .NET). Be sure to check the &amp;quot;Add action methods for Create, Update, Detail,
and Delete scenarios&amp;quot; as we will be using them later. 
&lt;/li&gt;
&lt;li&gt;
We now have a basic controller, but it doesn't do anything yet. First, we want to
modify our Index action, as it is the default action in the controller. We will use
this action to list all of the Employees that currently exist in our database. Your
Index() action method should contain the following code: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
return View(_entities.Employee.ToList());&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Save all and compile. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;h4&gt;Creating a View
&lt;/h4&gt;
&lt;p&gt;
We have our Model and we have our Controller, so it is time for our View. The View
is responsible for display only—it should contain virtually no logic. Our controller
doesn't do anything yet, but we can at least get the base file structure and navigation
in place. Since our Model governs &lt;em&gt;how&lt;/em&gt; and our Controller governs &lt;em&gt;what&lt;/em&gt;,
think of the View as governing &lt;em&gt;where&lt;/em&gt;, as it is responsible for deciding where
each data element gets displayed on your page.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Once you have saved and compiled your Controller, right-click on the Index action
name and select Add View. 
&lt;/li&gt;
&lt;li&gt;
The View should be named Index, just like your Controller action. Also, make your
View strongly typed to the Employee model. Since a list of Employees is being passed
to the View from the Controller, making the View strongly-typed prevents us from having
to cast our View Model from object to Employee. Finally, since we are providing a
list of Employees, the View Content should be a List. 
&lt;/li&gt;
&lt;li&gt;
The &lt;em&gt;Employee&lt;/em&gt; folder should be automatically created under the &lt;em&gt;/Views/&lt;/em&gt; folder,
and the &lt;em&gt;Index.aspx&lt;/em&gt; View inside of this new &lt;em&gt;Employee&lt;/em&gt; folder. 
&lt;/li&gt;
&lt;li&gt;
The last thing we need to do is provide some navigation to this view. Open up &lt;em&gt;/Views/Shared/Site.Master&lt;/em&gt; and
add an ActionLink within the Menu Container section to the Index action of the Employee
controller. When you are done, the Menu Container should look like this: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;&amp;lt;ul id=&amp;quot;menu&amp;quot;&amp;gt;              
  &amp;lt;li&amp;gt;&amp;lt;%= Html.ActionLink(&amp;quot;Home&amp;quot;, &amp;quot;Index&amp;quot;, &amp;quot;Home&amp;quot;)%&amp;gt;&amp;lt;/li&amp;gt;
  &amp;lt;li&amp;gt;&amp;lt;%= Html.ActionLink(&amp;quot;About&amp;quot;, &amp;quot;About&amp;quot;, &amp;quot;Home&amp;quot;)%&amp;gt;&amp;lt;/li&amp;gt;
  &amp;lt;li&amp;gt;&amp;lt;%= Html.ActionLink(&amp;quot;Employees&amp;quot;, &amp;quot;Index&amp;quot;, &amp;quot;Employee&amp;quot;)%&amp;gt;&amp;lt;/li&amp;gt;
&amp;lt;/ul&amp;gt;&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Save all and run. When you navigate to your Employees link, you should get a list
of all employees currently in the database with an Edit, Details, and Delete links. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;blockquote&gt; 
&lt;h5&gt;New in ASP.NET MVC 2: Strongly Typed HTML Helpers
&lt;/h5&gt;
&lt;p&gt;
In the previous version of ASP.NET MVC, HTML helpers were simple generic classes.
Generated views were full of Magic Strings for each property in your model, such as
&amp;lt;%= Html.TextBox(&amp;quot;Name&amp;quot;) %&amp;gt;, opening the door for a fat-fingered property
name. MVC 2 includes strongly-typed HTML helpers on strongly-typed views. For form-based
views, use the new strongly-typed &amp;quot;For&amp;quot; methods, such as &amp;lt;%= Html.TextBoxFor(model
=&amp;gt; model.Name) %&amp;gt; or &amp;lt;%= Html.EditorFor(model =&amp;gt; model.Name) %&amp;gt; to
eliminate the risk of incorrectly entering a property name. For display fields, use
Html.DisplayFor() to provide similar benefits for your read-only data, including the
elimination of HTML encoding for each field.
&lt;/p&gt;
&lt;/blockquote&gt; 
&lt;h4&gt;Adding New Data
&lt;/h4&gt;
&lt;p&gt;
A list of employees is great, but we also need the ability to manipulate that data.
First, let's start with the ability to create new data.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Within Visual Studio, open &lt;em&gt;/Controllers/EmployeeController&lt;/em&gt; and navigate to
the POST Create action. POST is an HTTP verb associated with pushing data to the server
in the HTTP header, commonly associated with form submits. GET is the HTTP verb for
pure retrieval, commonly associated with a direct URL request, such as clicking a
link. In this case, the POST action can be identified by the [HttpPost] decoration. 
&lt;/li&gt;
&lt;li&gt;
The method arguments currently include only a FormCollection object that will contain
all of the header values associated with the POST. However, MVC is smart; it can automatically
transform this collection into a type-safe Model, based on the names of the header
variables (the identities of the HTML form inputs match the names of the Model's properties).
The one exception is the Id attribute, which is available in the Model but not populated
until after the object is saved to the database. To get strong typing, and avoid having
to manually cast or map form collection data to a new instance of our Model, change
the method signature to the following: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;public ActionResult Create([Bind(Exclude=&amp;quot;Id&amp;quot;)] Employee newEmployee)&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Now that we have a populated Model, we just need to save the Model to the database
and redirect back to the List when we are done. Do this by replacing the contents
of the action method with the following code: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;try
{
  if (!ModelState.IsValid) return View();

  var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
  _entities.AddToEmployee(employee);
  _entities.SaveChanges();
  return RedirectToAction(&amp;quot;Index&amp;quot;);
}
catch
{
  return View();
}&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Save and compile. 
&lt;/li&gt;
&lt;li&gt;
Now we must create the View for adding data. As we did with Index, we can create the
view by right-clicking the Action method and selecting Add View. The view content
should be Create. 
&lt;/li&gt;
&lt;li&gt;
The only modification for the Create view is the Id property. The generated code creates
an input for this property, but it is not needed since the column value is auto-populated
by the database when saving the entity. Remove this input and label from the form. 
&lt;/li&gt;
&lt;li&gt;
Save and run. You should now be able to add new records to the database. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;blockquote&gt; 
&lt;h5&gt;New in ASP.NET MVC2: Better Verb Attributes
&lt;/h5&gt;
&lt;p&gt;
In the first version of ASP.NET MVC, HTTP Verb conditions were placed on an Action
via the AcceptVerbsAttribute, such as the Create action's [AcceptVerbs(HttpVerbs.Post)].
In ASP.NET MVC 2, these attributes have been simplified with the introduction of the
HttpGetAttribute, HttpDeleteAttribute, HttpPostAttribute, and HttpPutAttribute.
&lt;/p&gt;
&lt;/blockquote&gt; 
&lt;h4&gt;Routing and Updates
&lt;/h4&gt;
&lt;p&gt;
The end user can view a list of Employees, and can create new employees, but when
the end user clicks the Edit or Detail links, they get an error since these Views
haven't been created yet and the Actions are not implemented. One by one, we will
get the new views in place.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Within Visual Studio, open &lt;em&gt;/Controllers/EmployeeController&lt;/em&gt; and navigate to
the Details action. You may notice that the method already accepts an integer as input,
and shows example usage of the action in a commented URL above the method: &lt;em&gt;/Employee/Details/5&lt;/em&gt;.
This integer is the identity value of the Employee record, and is already populated
in the links of our List view created in the previous section. 
&lt;/li&gt;
&lt;li&gt;
Within Visual Studio, open &lt;em&gt;Global.asax&lt;/em&gt; and navigate to the RegisterRoutes
method. The default route for ASP.NET MVC is &lt;em&gt;/{controller}/{action}/{id}/&lt;/em&gt;.
By parsing out any URL into the application, MVC can determine which Controller to
use, which Action to execute, and which arguments to pass to the Action. Later portions
of the URL path are often optional, and when not specified, are replaced with the
default values: Home, Index, and String.Empty. The URLs of &amp;quot;/&amp;quot;, &amp;quot;/Home&amp;quot;,
&amp;quot;/Home/&amp;quot;, &amp;quot;/Home/Index&amp;quot;, and &amp;quot;/Home/Index/&amp;quot; are all
equivalent URLs in the eyes of ASP.NET MVC. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;
Go back to the Employee controller. Now that we know what the integer argument is
for, we need to retrieve the Employee matching the associated identity and pass it
to a view for editing or display.
&lt;/p&gt;
&lt;ol&gt;
&lt;li&gt;
Replace the contents of the GET version of the Edit action method with the following
code to retrieve the Employee from the database that matches the identity specified
in the route: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
return View(_entities.Employee.Where(emp =&amp;gt; emp.Id == id).First());&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Save and compile. 
&lt;/li&gt;
&lt;li&gt;
Right-click the Action and add the View. The View should still be strongly typed,
but this time the view content should be Edit. 
&lt;/li&gt;
&lt;li&gt;
The Details and GET Delete actions are largely similar as the GET Edit action, except
that the view is labels instead of text boxes. Repeat the above three steps for the
Details action method with a Details view content and for the GET Delete action method
with a Delete view content. 
&lt;/li&gt;
&lt;li&gt;
As with the Create view, the Id property should be removed from the Edit view, as
it is not an item that should be edited by the end user. 
&lt;/li&gt;
&lt;li&gt;
Save and run. You should see the details of an Employee when clicking on the Edit
and Details links. 
&lt;/li&gt;
&lt;li&gt;
We can view Employee details within the Edit form, but when we make changes and submit,
nothing happens. We need to modify the POST Edit action to save our changes back to
the database. The default POST Edit action accepts an id and a FormCollection as input
arguments, but similarly to the POST Create action, we can change this to use our
strongly typed model to avoid having to cast or map data. However, unlike our Create
action, we need to bind the id property so that the system knows which record to update.
To make these modifications, replace the POST Edit signature with the following: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;public ActionResult Edit(MvcJamSession.Models.Employee employee)&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Replace the contents of the POST Edit action method with the following code to save
the changes to the database: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;try
{
  if (!ModelState.IsValid) return View();

  var _entities = new MvcJamSession.Models.MvcJamSessionEntities();
  var _originalEmployee =
    _entities.Employee.Where(emp =&amp;gt; emp.Id == employee.Id).First();
  _entities.ApplyPropertyChanges(_originalEmployee.EntityKey.EntitySetName,
                                 employee);
  _entities.SaveChanges();
  return RedirectToAction(&amp;quot;Index&amp;quot;);
}
catch
{
  return View();
}&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Replace the signature of the POST Delete action method with the following code to
provide strong typing on our model: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;public ActionResult Delete(MvcJamSession.Models.Employee employee)&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Finally, replace the contents of the POST Delete action method with the following
code to delete a record: 
&lt;br /&gt;
&lt;pre class="csharp:nocontrols" name="code"&gt;var _entites = new MvcJamSession.Models.MvcJamSessionEntities();
var originalEmployee =
  _entites.Employee.Where(emp =&amp;gt; emp.Id == deletedEmployee.Id).First();
_entites.DeleteObject(originalEmployee);
_entites.SaveChanges();
return RedirectToAction(&amp;quot;Index&amp;quot;);&lt;/pre&gt;
&lt;/li&gt;
&lt;li&gt;
Save, compile, and run. You should now be able to modify existing Employee records. 
&lt;/li&gt;
&lt;/ol&gt;
&lt;p&gt;
We now have a fully-functional ASP.NET MVC application to manage our Employee records.
Congratulations!
&lt;/p&gt;
&lt;div style="padding-bottom: 0px; margin: 0px; padding-left: 0px; padding-right: 0px; display: inline; float: none; padding-top: 0px" id="scid:0767317B-992E-4b12-91E0-4F059A8CECA8:13d5b303-4a7f-4ac9-93a3-f208ffe24fe0" class="wlWriterEditableSmartContent"&gt;Technorati
Tags: &lt;a href="http://technorati.com/tags/ASP.NET+MVC" rel="tag"&gt;ASP.NET MVC&lt;/a&gt;,&lt;a href="http://technorati.com/tags/MVC" rel="tag"&gt;MVC&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Jam+Session" rel="tag"&gt;Jam
Session&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Coding+Exercise" rel="tag"&gt;Coding
Exercise&lt;/a&gt;,&lt;a href="http://technorati.com/tags/Learn+to+Code" rel="tag"&gt;Learn to
Code&lt;/a&gt;
&lt;/div&gt;
&lt;img width="0" height="0" src="http://www.cptloadtest.com/aggbug.ashx?id=1d5d9618-a51a-471f-a5ec-7381c916f11d" /&gt;</description>
      <comments>http://www.cptloadtest.com/CommentView,guid,1d5d9618-a51a-471f-a5ec-7381c916f11d.aspx</comments>
      <category>ASP.Net</category>
      <category>Learn to Code</category>
      <category>MVC</category>
    </item>
  </channel>
</rss>