Friday, February 11, 2011

Web User Control

Introduction
Let's create a simple calendar control using web user control.


Starting with user controls
So we decided to write our own control for selecting month/year pair. Then we will use it on our page.
Let's start by creating new web-site in Visual Studio 2005 and adding new Web user control to it.

Web user control consists of two parts: .ascx file, which is face of our control and contains HTML and ASP.NET 2.0 tags and code-behind partial class.
Now let's create a simple design for our control in .ascx file. Let's start by creating div, and placing a table inside of it. We will have 4 rows and 3 columns in our table (for 12 month).
<div style="border: 1px solid black; width: 100px; text-align: center;">
<table width=100% align=center>
      <tr>
            <td>Jan</td>
            <td>Feb</td>
            <td>Mar</td>
      </tr>
      <tr>
            <td>Apr</td>
            <td>May</td>
            <td>Jun</td>
      </tr>
      <tr>
            <td>Jul</td>
            <td>Aug</td>
            <td>Sep</td>
      </tr>
      <tr>
            <td>Oct</td>
            <td>Nov</td>
            <td>Dec</td>
      </tr>
</table>
</div>
Let's make a field for year input and place it on top of our table in separate div.
<div style="background-color: rgb(255,238,187); padding-top: 2px;">
<asp:TextBox ID=txtYear Runat=server style="width: 50px; border: 1px solid; text-align: center;" MaxLength ="4"></ asp: TextBox>
 <hr style="border: black 1px solid;" size=1>
</div>
For a simple navigation of year let's create a simple JavaScript function to skip to next or previous year.
To validate date we will use regular expressions in JavaScript. It's really ease to use regular expressions - you just need to provide a patter and tell RegExp class to test your input value against pattern. Here is JavaScript function which will do all this for us:
<script>
function nextYear(forward) {
      var yearField = document.getElementById("<%= txtYear.ClientID %>");
      var testEr = new RegExp('\\d{4}', 'g');
      if(testEr.test(yearField.value)) {
            var yearValue = parseInt(yearField.value);
            if(forward)
                  yearValue = yearValue+1;
            else
                  yearValue = yearValue-1;
            yearField.value = yearValue;
      } else {
      var newDate = new Date();
      yearField.value = newDate.getFullYear();
      }
}
</script
>
First of all it's getElementById which is standard JavaScript function. But if you look at its body you will notice the use of ASP.NET script as the parameter to getElementById function. ASP.NET renders its controls in respective HTML tags (and even several HTML tags) and gives them unique IDs. To ensure uniqueness ASP.NET may modify string provided in ID attribute of each ASP.NET control, adding this control id or page id. So our month calendar's text box on different pages can have different ids. That's why we use txtYear.ClientID, which returns the actual id of rendered input field.
Next thing worth to have a look at, is a RegExp. RegExp stands for Regular Expression and allows performing several operations (replace, split, test, etc) on input strings using patterns. Pattern defines the format of the string. In our case \d ("\\d" in JavaScript, because "\" is escape sequence) is built-in class, which denotes numbers. Number 4 in {4} means that \d class should appear in text exactly 4 times. This expression can be rewritten to several other forms (for example [0-9][0-9][0-9][0-9] or \d\d\d\d or [0-9]{4} etc).
If year in the input is correct, we then increase or decrease it by 1. But if test was not passed we simply get the current year (newDate.getFullYear()).
Next step is binding script we wrote to some simple control. I choose to use simple left-right-arrow images. Now let's modify our header div (where the year text box is) to make it look like:
<div style="background-color: rgb(255,238,187); padding-top: 2px;">
<img src="<%= Page.ResolveUrl("~/images/prev1.gif") %>" border=0 onclick="nextYear(false);">
<asp:TextBox ID=txtYear Runat=server style="width: 50px; border: 1px solid; text-align: center;" MaxLength="4"></asp:TextBox>
<img src="<%= Page.ResolveUrl("~/images/next1.gif") %>" border=0 onclick="nextYear(true);">
 <hr style="border: black 1px solid;" size=1>
</div>
Now our text box is bound to our script by means of small arrows which allow user to go back and forth in year selection.
Making months work
Now we approach to core functionality of our control. Now we should make months which are just static text for now to be ASP.NET link button control.
Replace month's table with the following one:
<table width=100% align=center>
      <tr>
<td><asp:LinkButton ID="lnkJan" Runat=server OnClick="lnkMonthClick>Jan</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkFeb" Runat=server
OnClick="lnkMonthClick>Feb</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkMar" Runat=server
OnClick="lnkMonthClick>Mar</asp:LinkButton>
            </td>
      </tr>
      <tr>
            <td><asp:LinkButton ID="lnkApr" Runat=server
OnClick="lnkMonthClick>Apr</asp:LinkButton>
            </td>
      <td><asp:LinkButton ID="lnkMay" Runat=server
OnClick="lnkMonthClick>May</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkJun" Runat=server
OnClick="lnkMonthClick>Jun</asp:LinkButton>
            </td>
      </tr>
      <tr>
<td><asp:LinkButton ID="lnkJul" Runat=server OnClick="lnkMonthClick>Jul</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkAug" Runat=server
OnClick="lnkMonthClick>Aug</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkSep" Runat=server
OnClick="lnkMonthClick>Sep</asp:LinkButton>
            </td>
      </tr>
      <tr>
            <td><asp:LinkButton ID="lnkOct" Runat=server
OnClick="lnkMonthClick>Oct</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkNov" Runat=server
            OnClick="lnkMonthClick>Nov</asp:LinkButton>
            </td>
            <td><asp:LinkButton ID="lnkDec" Runat=server
OnClick="lnkMonthClick>Dec</asp:LinkButton>
            </td>
      </tr>
</
table>
As you can see I wrapped month names in Link button and added event handler to all links. Now our work with .ascx file is done it's time to switch to code-behind class. In code behind we should first of all set the initial date of calendar and provide a value field so that pages and other controls can retrieve value from month calendar. Let's create a field and a property to hold value:
private DateTime _Value = DateTime.Now;
public DateTime Value
{
get { return _Value; }
      set { _Value = value; }
}
Next step is to create even handler to our source. Paste next fragment in code-behind class.
public void lnkMonthClick(object sender, EventArgs e)
{
int Month = 1;
      LinkButton btn = (LinkButton)sender;
      if (btn.Text == "Jan")Month = 1;
      if (btn.Text == "Feb")Month = 2;
      if (btn.Text == "Mar")Month = 3;
      if (btn.Text == "Apr")Month = 4;
      if (btn.Text == "May")Month = 5;
      if (btn.Text == "Jun")Month = 6;
      if (btn.Text == "Jul")Month = 7;
      if (btn.Text == "Aug")Month = 8;
      if (btn.Text == "Sep")Month = 9;
      if (btn.Text == "Oct")Month = 10;
      if (btn.Text == "Nov")Month = 11;
      if (btn.Text == "Dec")Month = 12;

      int Year;
      try {
      Year = Int32.Parse(txtYear.Text);
      } catch { Year = DateTime.Now.Year; }
      this.Value = new DateTime(Year, Month, 1, 0, 0, 1);
      if (DateChanged != null)
      DateChanged.Invoke(this, new EventArgs());
}
public event EventHandler DateChanged;
Using Month select control
Now let's add control to our Default.aspx page. Just drag-n-drop your month select control file from right panel to Default.aspx content pane. You should see something like this:

Now we should register event for month change. To do this, simply add this line to Page_Load:
MonthSelect1.DateChanged += new EventHandler(MonthSelect1_DateChanged);
And then add event handler code. For example:
void MonthSelect1_DateChanged(object sender, EventArgs e)
{
Label1.Text = MonthSelect1.Value.ToString();
}
 

No comments:

Post a Comment