<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>phdesign &#187; programming</title>
	<atom:link href="http://www.phdesign.com.au/category/programming/feed/" rel="self" type="application/rss+xml" />
	<link>http://www.phdesign.com.au</link>
	<description>phdesign web and application development by Paul Heasley</description>
	<lastBuildDate>Thu, 08 Jul 2010 00:08:54 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.0</generator>
		<item>
		<title>WPF RadioButton Binding to IsChecked Property</title>
		<link>http://www.phdesign.com.au/programming/wpf-radiobutton-binding-to-ischecked-property/</link>
		<comments>http://www.phdesign.com.au/programming/wpf-radiobutton-binding-to-ischecked-property/#comments</comments>
		<pubDate>Thu, 08 Jul 2010 00:08:30 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>
		<category><![CDATA[dotnet]]></category>
		<category><![CDATA[howto]]></category>
		<category><![CDATA[wpf]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/?p=226</guid>
		<description><![CDATA[I wasted a few development hours discovering this strange behaviour when binding to the IsChecked property of a RadioButton in WPF when using MVVM. You can read about the issue on the MSDN forum, but from my experience if you have a two way binding on the is IsChecked property of a RadioButton which is part [...]]]></description>
			<content:encoded><![CDATA[<p>I wasted a few development hours discovering this strange behaviour when binding to the IsChecked property of a RadioButton in WPF when using MVVM. You can <a title="RadioButton unchecked bindings issue still not resolved?" href="http://social.msdn.microsoft.com/forums/en-US/wpf/thread/8eb8280a-19c4-4502-8260-f74633a9e2f2/">read about the issue on the MSDN forum</a>, but from my experience if you have a two way binding on the is IsChecked property of a RadioButton which is part of a group, then after you set the bound property in code a couple of times, the RadioButton loses it&#8217;s binding all together.<span id="more-226"></span></p>
<p>Here is my original code that kept losing it&#8217;s binding:</p>
<pre><code>&lt;ItemsControl ItemsSource="{Binding Elements}" Margin="0,10,0,10"&gt;
  &lt;ItemsControl.ItemTemplate&gt;
    &lt;DataTemplate&gt;
      &lt;RadioButton Content="{Binding Key}"
        GroupName="{Binding RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}, Path=DataContext.Title}"
        IsChecked="{Binding Path=IsSelected}"
        Margin="0,5,0,5"/&gt;
    &lt;/DataTemplate&gt;
  &lt;/ItemsControl.ItemTemplate&gt;
&lt;/ItemsControl&gt;</code></pre>
<p>There a couple of ways to solve this,</p>
<ol>
<li><a title="Binding IsChecked property of RadioButton in WPF" href="http://pstaev.blogspot.com/2008/10/binding-ischecked-property-of.html">You can subclass the RadioButton</a></li>
<li><a title="Binding IsChecked property of RadioButton in WPF" href="http://pstaev.blogspot.com/2008/10/binding-ischecked-property-of.html"></a>You could use OneWay binding on IsChecked to set the RadioButton state from code, then bind the Command property to a  Command when user clicks on the RadioButton.
<pre><code>&lt;ItemsControl ItemsSource="{Binding Elements}" Margin="0,10,0,10"&gt;
  &lt;ItemsControl.ItemTemplate&gt;
    &lt;DataTemplate&gt;
      &lt;RadioButton Content="{Binding Key}"
        GroupName="{Binding RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}, Path=DataContext.Title}"
        IsChecked="{Binding Path=IsSelected, Mode=OneWay}"
        Command="{Binding IsSelectedCommand}"
        CommandParameter="{Binding}"
        Margin="0,5,0,5"/&gt;
    &lt;/DataTemplate&gt;
  &lt;/ItemsControl.ItemTemplate&gt;
&lt;/ItemsControl&gt;</code></pre>
</li>
<li>My preferred option: (<a title="RadioButton unchecked bindings issue still not resolved?" href="http://social.msdn.microsoft.com/forums/en-US/wpf/thread/8eb8280a-19c4-4502-8260-f74633a9e2f2/">as suggested by karliwatson</a>) use a ListBox for the grouping and bind the ListBoxItem&#8217;s IsSelected property to your ViewModel, then bind the RadioButton&#8217;s IsChecked property to the list box item&#8217;s IsSelected property. Don&#8217;t set a group on the RadioButton.
<pre><code>&lt;ListBox ItemsSource="{Binding Elements}" Margin="0,10,0,10" BorderThickness="0" Background="Transparent"&gt;
  &lt;ListBox.ItemContainerStyle&gt;
    &lt;Style TargetType="{x:Type ListBoxItem}"&gt;
      &lt;Setter Property="IsSelected" Value="{Binding Path=IsSelected, Mode=TwoWay}" /&gt;
      &lt;Setter Property="Template"&gt;
        &lt;Setter.Value&gt;
          &lt;ControlTemplate TargetType="ListBoxItem"&gt;
            &lt;ContentPresenter/&gt;
          &lt;/ControlTemplate&gt;
        &lt;/Setter.Value&gt;
      &lt;/Setter&gt;
    &lt;/Style&gt;
  &lt;/ListBox.ItemContainerStyle&gt;
  &lt;ListBox.ItemTemplate&gt;
    &lt;DataTemplate&gt;
        &lt;RadioButton Content="{Binding Key}"
            IsChecked="{Binding Path=IsSelected, Mode=TwoWay, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type ListBoxItem}}}"
            Margin="0,5,0,5"/&gt;
    &lt;/DataTemplate&gt;
  &lt;/ListBox.ItemTemplate&gt;
&lt;/ListBox&gt;</code></pre>
</li>
</ol>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/wpf-radiobutton-binding-to-ischecked-property/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Logging Messages in Dot Net</title>
		<link>http://www.phdesign.com.au/programming/logging-messages-in-dot-net/</link>
		<comments>http://www.phdesign.com.au/programming/logging-messages-in-dot-net/#comments</comments>
		<pubDate>Thu, 17 Jun 2010 10:39:15 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/?p=215</guid>
		<description><![CDATA[The .Net Framework provides an easy way to log messages using the System.Diagnostics.Trace class. By using Trace the user can configure where and what to log in the config file. Below are two simple ways to log a message to a text file. 1. The simplest way if you don&#8217;t care about the format of [...]]]></description>
			<content:encoded><![CDATA[<p>The .Net Framework provides an easy way to log messages using the System.Diagnostics.Trace class. By using Trace the user can configure where and what to log in the config file.</p>
<p>Below are two simple ways to log a message to a text file.<span id="more-215"></span></p>
<p>1. The simplest way if you don&#8217;t care about the format of the message is to use a filter on the trace listener in app.config to determine which events to log.</p>
<pre><code>&lt;system.diagnostics&gt;
  &lt;trace autoflush="true"&gt;
    &lt;listeners&gt;
      &lt;add name="LogListener" type="System.Diagnostics.TextWriterTraceListener" initializeData="trace.log" traceOutputOptions="DateTime"&gt;
        &lt;!--
        EventTypes:
        Critical: Fatal error or application crash.
        Error: Recoverable error.
        Warning: Noncritical problem.
        Information: Informational message.
        Verbose: Debugging trace.
        Start: Starting of a logical operation.
        Stop: Stopping of a logical operation.
        Suspend: Suspension of a logical operation.
        Resume: Resumption of a logical operation.
        Transfer: Changing of correlation identity.
        --&gt;
        &lt;filter type="System.Diagnostics.EventTypeFilter" initializeData="Error"/&gt;
      &lt;/add&gt;
    &lt;/listeners&gt;
  &lt;/trace&gt;
&lt;/system.diagnostics&gt;</code></pre>
<p>Then you only need to call one of the following methods to log a message with a specific EventType.</p>
<pre><code>System.Diagnostics.Trace.TraceInformation("Information messge");
System.Diagnostics.Trace.TraceError("Error messge");</code></pre>
<p>2. If you want more control over the format of the message you can use a TraceSwitch in app.config:</p>
<pre><code>&lt;system.diagnostics&gt;
  &lt;switches&gt;
    &lt;clear/&gt;
    &lt;!--
    Trace Levels:
    Off: 0: None
    Error: 1: Only error messages
    Warning: 2: Warning messages and error messages
    Info: 3: Informational messages, warning messages, and error messages
    Verbose: 4: Verbose messages, informational messages, warning messages, and error messages
    --&gt;
    &lt;add name="LogLevel" value="Info"/&gt;
  &lt;/switches&gt;
  &lt;trace autoflush="true"&gt;
    &lt;listeners&gt;
      &lt;add name="LogListener" type="System.Diagnostics.TextWriterTraceListener" initializeData="trace.log"/&gt;
    &lt;/listeners&gt;
  &lt;/trace&gt;
&lt;/system.diagnostics&gt;</code></pre>
<p>Then retrieve the TraceSwitch in code as follows:</p>
<pre><code>TraceSwitch logLevel = new TraceSwitch("LogLevel", "The logging level");
Trace.WriteLineIf(logLevel.Level &gt;= level, string.Format("[{0}] [{1}]: {2}", DateTime.Now, level, message));</code></pre>
<p>The examples are for a desktop application, but they would work equally well in a web application by configuring web.config. Note that web applications aren&#8217;t compiled with Trace enabled, so you need to enable them in web.config by using the following after the &lt;system.diagnostics&gt; node:</p>
<pre><code>&lt;system.codedom&gt;
  &lt;compilers&gt;
    &lt;compiler language="c#;cs;csharp"
      extension=".cs"
      compilerOptions="/d:TRACE"
      type="Microsoft.CSharp.CSharpCodeProvider, System, Version=2.0.3500.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" warningLevel="1" /&gt;
    &lt;/compilers&gt;
&lt;/system.codedom&gt;</code></pre>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/logging-messages-in-dot-net/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>IE Autocomplete Doesn&#8217;t Fire onchange Event Handler</title>
		<link>http://www.phdesign.com.au/programming/ie-autocomplete-doesnt-fire-onchange-event-handler/</link>
		<comments>http://www.phdesign.com.au/programming/ie-autocomplete-doesnt-fire-onchange-event-handler/#comments</comments>
		<pubDate>Sat, 17 Apr 2010 00:10:24 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>
		<category><![CDATA[howto]]></category>
		<category><![CDATA[javascript]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/?p=177</guid>
		<description><![CDATA[While testing an ASP validation control in Internet Explorer I found that the validation wasn&#8217;t triggered if I selected an item from the autocomplete list for a text box, this was the same in Safari for Windows, but worked fine in Mozilla Firefox. What?! Turns out that when selecting an item from the autocomplete list [...]]]></description>
			<content:encoded><![CDATA[<p>While testing an ASP validation control in Internet Explorer I found that the validation wasn&#8217;t triggered if I selected an item from the autocomplete list for a text box, this was the same in Safari for Windows, but worked fine in Mozilla Firefox. What?!</p>
<p><a href="http://www.phdesign.com.au/wp-content/uploads/2010/04/autocomplete.png"><img class="alignnone size-full wp-image-185" title="IE form autocomplete feature" src="http://www.phdesign.com.au/wp-content/uploads/2010/04/autocomplete.png" alt="IE form autocomplete feature" width="152" height="66" /></a></p>
<p><span id="more-177"></span>Turns out that when selecting an item from the autocomplete list the onchange event handler doesn&#8217;t fire, therefore the validation scripts don&#8217;t fire. The onblur event does fire however, meaning we can do a check to see if the data has changed and if so, manually fire the onchange event handler.</p>
<pre><code>/*
Author:     Paul Heasley
Date:       15.04.2010
Version:    0.1
Description:
IE and safari do not trigger the onchange events for textboxes when using autocomplete,
so validation events don't trigger. The blur event does trigger however so this script
checks to see if the value has changed (by saving the value on focus) and manually
calling onchange.
*/
function onChangeFix() {
    var previousValue = [];
    // Get all input elements
    var inputs = document.getElementsByTagName('input');

    for (var i = 0; i &lt; inputs.length; i++) {
        var elt = inputs[i];

        // Only update text boxes. Depending on your application you
        // may also need to fix text areas.
        if (elt.type.toLowerCase() == "text")
        {
            // Save old value.
            addEvent(elt, 'focus', function() {
                previousValue[this] = this.value;
            });

            // Compare to old value, do we need to trigger an onchange event?
            addEvent(elt, 'blur', function() {
                if (previousValue[this] != this.value)
                {
                    if (this.onchange)
                        this.onchange();
                }
            });

            // Set old value = new value to stop the blur event triggering
            // another onchange.
            addEvent(elt, 'change', function() {
                previousValue[this] = this.value;
            });
        }
    }
}</code></pre>
<p>Alternatively, you can disable the autocomplete feature by adding the attrribute autocomplete=&#8221;off&#8221; for either a specific input or for the entire form.</p>
<pre><code>&lt;form id="theForm" name="theForm" method="post" autocomplete="off"&gt;</code></pre>
<p>Download the entire code sample including the addEvent() function.</p>
<p><a title="Download the source" href="http://www.phdesign.com.au/wp-content/uploads/2010/04/autocomplete_fix.js"><img class="alignnone size-full wp-image-126" src="http://www.phdesign.com.au/wp-content/uploads/2009/11/button_download_source.gif" alt="Download the source" width="149" height="25" /></a></p>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/ie-autocomplete-doesnt-fire-onchange-event-handler/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Registering phreplace DLL in Windows Vista</title>
		<link>http://www.phdesign.com.au/programming/registering-phreplace-dll-in-windows-vista/</link>
		<comments>http://www.phdesign.com.au/programming/registering-phreplace-dll-in-windows-vista/#comments</comments>
		<pubDate>Wed, 17 Mar 2010 10:10:46 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>
		<category><![CDATA[dll]]></category>
		<category><![CDATA[phreplace]]></category>
		<category><![CDATA[vb6]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/?p=165</guid>
		<description><![CDATA[is a VB6 DLL which is called from a vbscript plugin for PSPad. The script tests to see if the correct version of the DLL is registered on this computer, if not it registers it using regsvr32. This works fine in XP but not Vista where the script will error. A while ago a user [...]]]></description>
			<content:encoded><![CDATA[<p><a href="http://www.phdesign.com.au/phreplace/" title="phreplace Multiline search / replace for PSPad">phreplace</a> is a VB6 DLL which is called from a vbscript plugin for <a title="PSPad editor for developers" href="http://www.pspad.com/">PSPad</a>. The script tests to see if the correct version of the DLL is registered on this computer, if not it registers it using regsvr32. This works fine in XP but not Vista where the script will error.</p>
<p>A while ago a user pointed out to me <a title="Self Registering DLL's" href="http://www.experts-exchange.com/Programming/System/Windows__Programming/Q_21243930.html">this article on Experts Exchange</a> explaining why administrative rights are required to register a DLL in Windows Vista, in the hopes that we could automate the registration of phreplace in Vista.<span id="more-165"></span></p>
<p>Feeling defeated, I decided to explain why I haven&#8217;t been able to do this yet.</p>
<p>The fundamental problem is that Vista requires administrative privledges to write to the registry key required to register DLLs:</p>
<p>HKEY_CLASSES_ROOT\</p>
<p>However the user can register DLLs in their own section of the registry:</p>
<p>HKEY_CURRENT_USER\SOFTWARE\Classes\</p>
<p>So all is well right? Well not if you&#8217;re using VB6, you see when you run regsvr32, it calls the methods DllRegisterSever and DllUnregisterSever  to add / remove entries from the registry, these methods are generated automatically by VB6 to register to HKEY_CLASSES_ROOT, and there is no way to change this.</p>
<p>Well, there is one way. Apparently the <a title="vbAdvance Visual Basic addon" href="http://vb.mvps.org/tools/vbAdvance/">vbAdvance</a> addon for VB6 allowed a user to customise DllRegisterServer / DllUnregisterServer, this sounds great, except that it doesn&#8217;t work. I tried modifying these methods according to the instruction, but the resulting DLL just wont register. I even tried registering the included example DLL in the download package, but this threw the same error.</p>
<p>I&#8217;m certain it must have worked at some stage, they wouldn&#8217;t have included an example that didn&#8217;t work, so I can&#8217;t figure out what&#8217;s different. Did it only work in Windows 98 or 2000?</p>
<p>So in summary, for as long as I&#8217;m using VB6 I don&#8217;t think I can change where phreplace tries to register the DLL, the only other workaround is to use RegOverridePredefKey to tell the registry to redirect any calls to HKEY_CLASSES_ROOT to HKEY_CURRENT_USER\SOFTWARE\Classes, this would work but I don&#8217;t think it can be done with vbscript, so I would need to add another executable to phreplace to use in registering / unregistering the DLL from the registry, seems pretty messy.</p>
<p>If you&#8217;ve got any tips for me (especially how I can customise DllRegisterServer) feel free to email me. I haven&#8217;t setup the facility for leaving comments on this site yet, sorry.</p>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/registering-phreplace-dll-in-windows-vista/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>No Validation Messages when Disabling &amp; Enabling ASP.Net Validator in Javascript</title>
		<link>http://www.phdesign.com.au/programming/no-validation-messages-when-disabling-enabling-asp-net-validator-in-javascript/</link>
		<comments>http://www.phdesign.com.au/programming/no-validation-messages-when-disabling-enabling-asp-net-validator-in-javascript/#comments</comments>
		<pubDate>Thu, 04 Mar 2010 06:21:31 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>
		<category><![CDATA[asp]]></category>
		<category><![CDATA[c#]]></category>
		<category><![CDATA[dotnet]]></category>
		<category><![CDATA[howto]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/?p=152</guid>
		<description><![CDATA[Need to validate a form field based on the value of another field in ASP.Net? The .Net Framework provides a javascript function to do just that: ValidatorEnable(val, enable) This function takes the validator control (val) and a boolean to determine whether it should be enable or disabled. The problem with this function is that it [...]]]></description>
			<content:encoded><![CDATA[<p>Need to validate a form field based on the value of another field in ASP.Net? The .Net Framework provides a javascript function to do just that:</p>
<pre><code>ValidatorEnable(val, enable)</code></pre>
<p>This function takes the validator control (val) and a boolean to determine whether it should be enable or disabled. The problem with this function is that it triggers the validator to validate itself immediately, showing any validation error messages before the user has submitted the form, which may not always be want you want.  The alternative is to set the validator&#8217;s enabled property to trigger this validator to validate itself when the user submits the form, and by using the Page_IsValid variable you can test whether the user has triggered a validation or not, then use ValidatorEnable function only after the validation has occurred when the user expects to see validation messages.<span id="more-152"></span></p>
<p>Consider a scenario where you want the user to enter their address only if they select Australia as the country, you also need to validate the address fields. Two ways to approach this are:</p>
<ol>
<li>For each address field you need to validate use a CustomValidator control, and in the ClientValidationFunction first check to see if the country is set to Australia and if it&#8217;s not then consider any response for this control as valid. The problem with this is you will have to recrete all of the validator functionality in your client validation function, which means more work.</li>
<li>Use standard validator controls for the address fields, and enable or disable them in javascript.  The code you call to enable or disable the validators could be a CustomValidation control on the country field, or a standard javascript call when the user selects a country. If the Page_IsValid variable is set to true then assume that the page has just been loaded and no validation has occured, in this case just set the validator controls enabled property so it will be validated (or not) when the user submits the form.<br />
If Page_IsValid is false then assume the user has tried to submit but failed, in this case we expect validation messages to show / hide as the user changes data on the form &#8211; this is the normal ASP.Net behaviour.</li>
</ol>
<p>Here&#8217;s an example:</p>
<pre><code>// The ClientValidationFunction of a custom validator for country
function validateCountry(source, args) {
    // We we haven't submitted the form yet
    if (Page_IsValid) {
        // Turn the postcode validator on / off, but don't validate
        requiredFieldValidatorPostcode.enabled = enable;
    } else {
        // Turn the postcode validator on / off, and immediately validate
        ValidatorEnable(requiredFieldValidatorPostcode, enable);
    }

    // Validate country is not empty
    args.IsValid = (args.Value.length &gt; 0);
}</code></pre>
<p>Don&#8217;t forget to include server side code to mimic this enabling / disabling of validators. To do this override the Validate() function, enable or disable validators depending on the country selected and call the base.Validate() function.</p>
<pre><code>public override void Validate()
{
    // Check if we need to validate the postcode.
    bool requireAddress = (dropDownListCountry.Text == "Australia");
    requiredFieldValidatorPostcode.Enabled = requireAddress;

    // Call the normal validation methods.
    base.Validate();
}

protected void customValidatorCountry_ServerValidate(object source, ServerValidateEventArgs args)
{
    // Implement the server side custom validation for country.
    args.IsValid = (args.Value.Length &gt; 0);
}</code></pre>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/no-validation-messages-when-disabling-enabling-asp-net-validator-in-javascript/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Microsoft Certified Professional Training Kit 70-536 Errata</title>
		<link>http://www.phdesign.com.au/programming/microsoft-certified-professional-training-kit-70-536-errata/</link>
		<comments>http://www.phdesign.com.au/programming/microsoft-certified-professional-training-kit-70-536-errata/#comments</comments>
		<pubDate>Sun, 10 Jan 2010 22:37:19 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>
		<category><![CDATA[c#]]></category>
		<category><![CDATA[dotnet]]></category>
		<category><![CDATA[microsoft]]></category>
		<category><![CDATA[training]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/?p=145</guid>
		<description><![CDATA[I&#8217;m currently studying for the Microsoft® .NET Framework 2.0—Application Development Foundation (70-536) exam, using the Microsoft Press Training Kit and I am appalled at the number of errors, both editorial and technical, in this book. There are whole paragraphs that have been taken out of one chapter and used as a place holder in another [...]]]></description>
			<content:encoded><![CDATA[<p>I&#8217;m currently studying for the Microsoft® .NET Framework 2.0—Application Development Foundation (70-536) exam, using the Microsoft Press Training Kit and I am appalled at the number of errors, both editorial and technical, in this book. There are whole paragraphs that have been taken out of one chapter and used as a place holder in another chapter then never changed, at one stage I was very confused when the book was talking about two different classes that seemed to do the same thing, only to find that they were the same class, it&#8217;s name had just be written wrong in places, arrggg.<span id="more-145"></span></p>
<p>The problem I find is when I&#8217;m not sure about something in the book, I don&#8217;t know whether my understanding is wrong or the book is wrong, so I went hunting for a list of known errors and sure enough, Microsoft already know about them, in fact there are so many they created 4 knowledge base pages for them! When combining these in a Word document it ended up being 90+ pages, that&#8217;s really bad.</p>
<p>If you have this book or think about getting it, here are the official list of updates from Microsoft:</p>
<p><a title="70-536 Comments and Corrections Part 1" href="http://support.microsoft.com/kb/923018" target="_blank">MCTS Self-Paced Training Kit (Exam 70-536): Microsoft® .NET Framework 2.0—Application Development Foundation comments and corrections Part 1</a></p>
<p><a title="70-536 Comments and Corrections Part 2" href="http://support.microsoft.com/kb/935218/" target="_blank">MCTS Self-Paced Training Kit (Exam 70-536): Microsoft® .NET Framework 2.0—Application Development Foundation Comments and Corrections Part 2</a></p>
<p><a title="70-536 Comments and Corrections Part 3" href="http://support.microsoft.com/kb/949730/" target="_blank">MCTS Self-Paced Training Kit (Exam 70-536): Microsoft® .NET Framework 2.0—Application Development Foundation Comments and Corrections Part 3</a></p>
<p><a title="70-536 Comments and Corrections Part 4" href="http://support.microsoft.com/kb/949734/" target="_blank">MCTS Self-Paced Training Kit (Exam 70-536): Microsoft® .NET Framework 2.0—Application Development Foundation Comments and Corrections Part 4</a></p>
<p>Perhaps it&#8217;s time to reprint the book?</p>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/microsoft-certified-professional-training-kit-70-536-errata/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Create a PDF using the .Net ReportViewer control</title>
		<link>http://www.phdesign.com.au/programming/create-a-pdf-using-the-net-reportviewer-control/</link>
		<comments>http://www.phdesign.com.au/programming/create-a-pdf-using-the-net-reportviewer-control/#comments</comments>
		<pubDate>Tue, 24 Nov 2009 23:48:48 +0000</pubDate>
		<dc:creator>paul</dc:creator>
				<category><![CDATA[programming]]></category>
		<category><![CDATA[asp]]></category>
		<category><![CDATA[c#]]></category>
		<category><![CDATA[dotnet]]></category>
		<category><![CDATA[howto]]></category>

		<guid isPermaLink="false">http://www.phdesign.com.au/apps/staging/?p=61</guid>
		<description><![CDATA[Need to generate a pdf programmatically in dot net? You could purchase an HTML to PDF converter and worry about getting the page to format nicely for paper, or you could use the freely available ReportViewer control from Microsoft. Microsoft offers SQL Server Reporting Services (SSRS) with SQL Server 2005 and later editions to generate [...]]]></description>
			<content:encoded><![CDATA[<p>Need to generate a pdf programmatically in dot net? You could purchase an HTML to PDF converter and worry about getting the page to format nicely for paper, or you could use the freely available ReportViewer control from Microsoft.</p>
<p>Microsoft offers SQL Server Reporting Services (SSRS) with SQL Server 2005 and later editions to generate and publish reports using the Report Definition Language (RDL), as an alternative to Crystal Reports. SSRS requires a Report Server to publish reports to so they can be useful, however they also offer a ASP.Net or Windows Forms component called the <a title="ReportViewer Controls on MSDN" href="http://msdn.microsoft.com/en-us/library/ms251671(VS.80).aspx">ReportViewer</a> which we can use in our code without the need for any SQL database to generate reports, and these reports can be saved as Excel files, PDF&#8217;s or images.<span id="more-61"></span></p>
<p>The ReportViewer uses a modified version of RDL called RDLC, which does not save any information about the data source along with the report, which makes it perfect for generating reports from objects that aren&#8217;t a database. For instance, I recently needed to generate PDF letters with custom information in them, so this solution proved perfect.</p>
<p>I&#8217;m going to demonstrate how to pass parameters to an ASP.Net page and return a PDF with that information in it, this could just as easily be done in a Windows Forms project. I&#8217;m using Visual Studio 2005, in my preferred language of c#.</p>
<p>Start by creating a new Web Site in Visual Studio. We are going to use a collection of custom business objects as the data source for our report, so add a new class to our project and call it Person.cs. Copy the following code into your Person.cs file.</p>
<pre><code>using System;
using System.Collections.Generic;

/// &lt;summary&gt;
/// The business object we want to use in our report.
/// &lt;/summary&gt;
public class Person
{
    /// &lt;summary&gt;
    /// Some fields to hold information about this person.
    /// &lt;/summary&gt;
    private string _title;
    private string _firstName;
    private string _lastName;

    /// &lt;summary&gt;
    /// The person properties are only able to be set at instantiation.
    /// After the object is created the properties are read only.
    /// &lt;/summary&gt;
    public string Title
    {
        get { return _title; }
    }

    public string FirstName
    {
        get { return _firstName; }
    }

    public string LastName
    {
        get { return _lastName; }
    }

    /// &lt;summary&gt;
    /// The person constructor.
    /// &lt;/summary&gt;
    /// &lt;param name="title"&gt;The person's title&lt;/param&gt;
    /// &lt;param name="firstName"&gt;Their first name&lt;/param&gt;
    /// &lt;param name="lastname"&gt;Their last name&lt;/param&gt;
    public Person(string title, string firstName, string lastname)
    {
        _title = title;
        _firstName = firstName;
        _lastName = lastname;
    }
}

/// &lt;summary&gt;
/// Creates a data source for our report.
/// &lt;/summary&gt;
public class PersonDataSource
{
    /// &lt;summary&gt;
    /// This method creates a new person object and adds them to a new collection that
    /// can be used as a data source for the report.
    /// &lt;/summary&gt;
    /// &lt;returns&gt;A collection of person objects.&lt;/returns&gt;
    public static List&lt;Person&gt; GetPerson(string title, string firstName, string lastName)
    {
        List&lt;Person&gt; people = new List&lt;Person&gt;();
        people.Add(new Person(title, firstName, lastName));
        return people;
    }
}</code></pre>
<p>Now that we have our business object we can create the report. Click add item on the project and select Report. Call this item Report.rdlc.</p>
<p>In the left hand toolbar you should see <em>Website Data Sources</em> (if not select <em>Data &gt; Show Data Sources</em> from the file menu) and in the data sources you should see our Person object. You can now build a report using the fields from our person object and the tools in the toobox. For simplicity just drag some Textboxes onto the report from the toolbox and enter any static text in them that you want, then drag on the Person&#8217;s title, firstName and lastName and arrange them appropriately.</p>
<p><img class="alignnone size-full wp-image-68" title="ReportViewer business object data source" src="http://www.phdesign.com.au/wp-content/uploads/2009/11/report_viewer_demo_1.jpg" alt="ReportViewer business object data source" width="251" height="126" /></p>
<p>Finally we need to write the code which will create the person data source and generate the report. Before we do this we need a reference to the ReportViewer control, right click on the project and <em>Add Reference</em>, in the .Net tab, select <em>Microsoft.ReportViewer.WebForms</em>.Now copy the following code into the Default.aspx source.</p>
<pre><code>&lt;%@ Page Language="C#" AutoEventWireup="true"  CodeFile="Default.aspx.cs" Inherits="_Default" %&gt;
&lt;!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"&gt;
&lt;html xmlns="http://www.w3.org/1999/xhtml" &gt;
&lt;head runat="server"&gt;
	&lt;title&gt;Untitled Page&lt;/title&gt;
&lt;/head&gt;
&lt;body&gt;
	&lt;form id="form1" runat="server"&gt;
	&lt;div&gt;
	&lt;asp:Label ID="lblTitle" runat="server" Text="Title: "&gt;&lt;/asp:Label&gt;
	&lt;asp:TextBox ID="txtTitle" runat="server"&gt;Mr&lt;/asp:TextBox&gt;&lt;br /&gt;
	&lt;asp:Label ID="lblFirstName" runat="server" Text="First Name: "&gt;&lt;/asp:Label&gt;
	&lt;asp:TextBox ID="txtFirstName" runat="server"&gt;John&lt;/asp:TextBox&gt;&lt;br /&gt;
	&lt;asp:Label ID="lblLastName" runat="server" Text="Last Name: "&gt;&lt;/asp:Label&gt;
	&lt;asp:TextBox ID="txtLastName" runat="server"&gt;Smith&lt;/asp:TextBox&gt;&lt;br /&gt;
	&lt;asp:Button ID="Button1" runat="server" Text="Submit" /&gt;&lt;/div&gt;
	&lt;/form&gt;
&lt;/body&gt;
&lt;/html&gt;</code></pre>
<p>And copy the following into the Code View of Default.aspx (Default.aspx.cs).</p>
<pre><code>using System;
using System.Web;
using Microsoft.Reporting.WebForms;
using System.IO;
using System.Text;
using System.Collections.Generic;
public partial class _Default : System.Web.UI.Page
{
	protected void Page_Load(object sender, EventArgs e)
	{
		if (Page.IsPostBack)
		{
			LocalReport report = new LocalReport();
			string reportName = "CompletionCertificate";
			string deviceInfo =
				"&lt;DeviceInfo&gt;" +
				"  &lt;OutputFormat&gt;EMF&lt;/OutputFormat&gt;" +
				"  &lt;PageWidth&gt;8.5in&lt;/PageWidth&gt;" +
				"  &lt;PageHeight&gt;11in&lt;/PageHeight&gt;" +
				"  &lt;MarginTop&gt;0.25in&lt;/MarginTop&gt;" +
				"  &lt;MarginLeft&gt;0.25in&lt;/MarginLeft&gt;" +
				"  &lt;MarginRight&gt;0.25in&lt;/MarginRight&gt;" +
				"  &lt;MarginBottom&gt;0.25in&lt;/MarginBottom&gt;" +
				"&lt;/DeviceInfo&gt;";
			Warning[] warnings;
			string[] streamids;
			string mimeType;
			string encoding;
			string extension;

			report.ReportPath = "Report.rdlc";
			report.DataSources.Add(new ReportDataSource("Person", PersonDataSource.GetPerson(txtTitle.Text, txtFirstName.Text, txtLastName.Text)));
			byte[] bytes = report.Render("PDF", deviceInfo, out mimeType, out encoding, out extension, out streamids, out warnings);

			Response.Clear();
			Response.ContentType = mimeType;
			Response.AddHeader("Content-Disposition", "attachment; filename=" + reportName + "." + extension);
			Response.OutputStream.Write(bytes, 0, bytes.Length);
			Response.End();
		}
	}
}</code></pre>
<p>Now run your project, enter some person data into the form and click submit. Open the PDF file and verify that the report is addressed to your person.</p>
<p><a href="http://www.phdesign.com.au/wp-content/uploads/2009/11/ReportViewerDemo.zip"><img class="alignnone size-full wp-image-126" title="Download the source for this tutorial" src="http://www.phdesign.com.au/wp-content/uploads/2009/11/button_download_source.gif" alt="Download the source for this tutorial" width="149" height="25" /></a></p>
]]></content:encoded>
			<wfw:commentRss>http://www.phdesign.com.au/programming/create-a-pdf-using-the-net-reportviewer-control/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
