Ramblings of a Software Engineer, Amusements of a Geek, Cacophony of a Guitarist, An Entropy Admirer's and an Interesting Character's Musings..
Wednesday, 24 November 2010
WPF Open Type Fonts!
for rendering the following in WFP.
ಕನ್ನಡ
हिंदी बे
గుల్తే ను
WPF 4.0, 3.5 adds Open Type Font Rendering support for a pletora of Indic Scripts. Check this out
WPF 3.0 had limited support.
Good work Microsoft!
A good read on WPF Globalization and Localization
Sunday, 4 October 2009
MVVM Pattern
The most useful pattern for WPF and Silverlight apps.
Wednesday, 22 October 2008
A Simple RSS Reader in WPF
I present here, a simple Rss reader written almost entirely in XAML. I personally find WPF Data binding powerful and exciting to work with.
To start with, i've used the XmlDataProvider class to hook up the source to an rss feed and set it's XPath property to "rss/channel/item", so that i get all items in the channel.
The layout is an utterly simple dockpanel which has a stackpanel docked to the top, a status bar docked to the bottom and a grid which fills up the remaining space.
The databinding is self explanatory with the following code.,However an important thing to note is that the <Textbox> which has the rss feed link is binded to the source directly. The 'UpdateSourceTrigger' property is by default set to LostFocus(). I've changed it to 'PropertyChanged', so that when a user enters an rss feed, the bindings get updated. Also notice that 'BindsDirectlyToSource' is set to True. Another interesting feature is the Master-Detail binding. Notice that between Line numbers 53-63. The listbox's selection is binded to description and link which are textblocks inside a stack panel. Now, when the listbox gets focus, we bind the frame's source to the uri in the texbox.
I've added some styles to the listbox aswell. all of which could be found in the source code. This surely is very simple and yet a foundation to write a powerful rss reader in wpf. :)
We could add more rss feeds, write value converters, validation rules etc. as required.
Window1.xaml
1: <Window x:Class="RssReader.Window1"
2: xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
3: xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
4: Title="RssReader" Height="600" Width="800">
5:
6: <Window.Resources>
7: <XmlDataProvider x:Key="rssdata"
8: Source="http://newsrss.bbc.co.uk/rss/newsonline_uk_edition/front_page/rss.xml"
9: XPath="rss/channel/item"/>
10:
11: </Window.Resources>
12:
13: <DockPanel DataContext="{Binding Source={StaticResource rssdata}}">
14: <StackPanel DockPanel.Dock="Top"
15: TextElement.FontWeight="Bold" Background="Gray">
16:
17: <TextBlock Text="{Binding XPath=./../title}"
18: FontSize="20" Margin="10 10 10 0"/>
19:
20: <TextBlock Text="{Binding XPath=./../description}"
21: FontSize="10" FontWeight="Normal" Margin="10 0"/>
22:
23: <TextBox Margin="5" Text="{Binding Source={StaticResource rssdata},
24: BindsDirectlyToSource=True,
25: Path=Source,
26: UpdateSourceTrigger=PropertyChanged}"/>
27: </StackPanel>
28:
29: <StatusBar DockPanel.Dock="Bottom">
30: <StatusBarItem Content="{Binding XPath=title}"/>
31: <Separator/>
32: <StatusBarItem Content="{Binding XPath=pubDate}"/>
33: </StatusBar>
34:
35: <Grid>
36: <Grid.ColumnDefinitions>
37: <ColumnDefinition Width="25*"/>
38: <ColumnDefinition Width="75*"/>
39: <ColumnDefinition/>
40: </Grid.ColumnDefinitions>
41:
42: <ListBox Grid.Column="0" IsSynchronizedWithCurrentItem="True"
43: ItemsSource="{Binding}" DisplayMemberPath="title"
44: Style="{StaticResource ListBoxHand}"/>
45: <GridSplitter/>
46:
47: <Grid Grid.Column="1">
48: <Grid.RowDefinitions>
49: <RowDefinition Height="Auto"> </RowDefinition>
50: <RowDefinition Height="85*"> </RowDefinition>
51: </Grid.RowDefinitions>
52:
53: <ListBox x:Name="selection" Grid.Row="0" IsSynchronizedWithCurrentItem="True"
54: Style="{StaticResource SimpleListBox}"
55: VerticalAlignment="Stretch"
56: GotFocus="selection_GotFocus">
57:
58: <StackPanel>
59: <TextBlock Text="{Binding XPath=description}" />
60: <TextBlock x:Name="txtlink" Text="{Binding XPath=link}"/>
61: </StackPanel>
62: </ListBox>
63: <Frame x:Name="Explorer" Grid.Row="1"/>
64: </Grid>
65: </Grid>
66:
67: </DockPanel>
68: </Window>
69:
Window1.xaml.cs
1: private void selection_GotFocus(object sender, RoutedEventArgs e) {
2: Uri uri = new Uri(this.txtlink.Text.ToString());
3: this.Explorer.Source = uri;
4: }
5:
Screenie
C0d3 : here
Tuesday, 7 October 2008
Winforms vs WPF
We have two powerful technologies to build windows client applications. The obvious questions are
when to use which technology ?
Which technology to choose for new applications and when organizations developing Windows Forms applications should adopt WPF ?
Answers here
Quick conclusions from the same source.
Monday, 6 October 2008
Compiling a Hello World WPF App
Getting to know the Binary Application Markup Language and the ‘.g.cs ‘files
XAML code gets compiled into a combination of C# file (.g.cs) and a BAML (.baml). The BAML file is an optimized, pre-tokenized version of the elements that were specified in the .xaml file. This is injected into the .Net Assembly (PE/DLL).
Notice that the static main method() gets generated in the App.g.cs file which points to an instance of WPF core application object (System.Windows.Application)
/// <summary>
/// Application Entry Point.
/// </summary>
[System.STAThreadAttribute()]
[System.Diagnostics.DebuggerNonUserCodeAttribute()]
public static void Main() {
HelloWorld.App app = new HelloWorld.App();
app.InitializeComponent();
app.Run();
}
The MS build project file has the <ApplicationDefinition> tag which is used to specify the xaml file that defines the application class.(app.xaml) Below is a cut and paste from my HelloWorld.csproject file.
<?xml version="1.0" encoding="utf-8"?>
<Project ToolsVersion="3.5" DefaultTargets="Build" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<Configuration Condition=" '$(Configuration)' == '' ">Debug</Configuration>
<Platform Condition=" '$(Platform)' == '' ">AnyCPU</Platform>
<ProductVersion>9.0.21022</ProductVersion>
<SchemaVersion>2.0</SchemaVersion>
<ProjectGuid>{15622594-2CE3-40D5-9D03-8A3F8AE7C3AB}</ProjectGuid>
<OutputType>WinExe</OutputType>
<AppDesignerFolder>Properties</AppDesignerFolder>
<RootNamespace>HelloWorld</RootNamespace>
<AssemblyName>HelloWorld</AssemblyName>
<TargetFrameworkVersion>v3.5</TargetFrameworkVersion>
<FileAlignment>512</FileAlignment>
<ProjectTypeGuids>{60dc8134-eba5-43b8-bcc9-bb4bc16c2548};{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}</ProjectTypeGuids>
<WarningLevel>4</WarningLevel>
<StartupObject>HelloWorld.App</StartupObject>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
<DebugSymbols>true</DebugSymbols>
<DebugType>full</DebugType>
<Optimize>false</Optimize>
<OutputPath>bin\Debug\</OutputPath>
<DefineConstants>DEBUG;TRACE</DefineConstants>
<ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Release|AnyCPU' ">
<DebugType>pdbonly</DebugType>
<Optimize>true</Optimize>
<OutputPath>bin\Release\</OutputPath>
<DefineConstants>TRACE</DefineConstants>
<ErrorReport>prompt</ErrorReport>
<WarningLevel>4</WarningLevel>
</PropertyGroup>
<ItemGroup>
<Reference Include="System" />
<Reference Include="System.Core">
<RequiredTargetFramework>3.5</RequiredTargetFramework>
</Reference>
<Reference Include="System.Xml.Linq">
<RequiredTargetFramework>3.5</RequiredTargetFramework>
</Reference>
<Reference Include="System.Data.DataSetExtensions">
<RequiredTargetFramework>3.5</RequiredTargetFramework>
</Reference>
<Reference Include="System.Data" />
<Reference Include="System.Xml" />
<Reference Include="WindowsBase" />
<Reference Include="PresentationCore" />
<Reference Include="PresentationFramework" />
</ItemGroup>
<ItemGroup>
<ApplicationDefinition Include="App.xaml">
<Generator>MSBuild:Compile</Generator>
<SubType>Designer</SubType>
</ApplicationDefinition>
<Page Include="Window1.xaml">
<Generator>MSBuild:Compile</Generator>
<SubType>Designer</SubType>
</Page>
<Compile Include="App.xaml.cs">
<DependentUpon>App.xaml</DependentUpon>
<SubType>Code</SubType>
</Compile>
<Compile Include="Window1.xaml.cs">
<DependentUpon>Window1.xaml</DependentUpon>
<SubType>Code</SubType>
</Compile>
</ItemGroup>
<ItemGroup>
<Compile Include="Properties\AssemblyInfo.cs">
<SubType>Code</SubType>
</Compile>
<Compile Include="Properties\Resources.Designer.cs">
<AutoGen>True</AutoGen>
<DesignTime>True</DesignTime>
<DependentUpon>Resources.resx</DependentUpon>
</Compile>
<Compile Include="Properties\Settings.Designer.cs">
<AutoGen>True</AutoGen>
<DependentUpon>Settings.settings</DependentUpon>
<DesignTimeSharedInput>True</DesignTimeSharedInput>
</Compile>
<Compile Include="Test.cs" />
<EmbeddedResource Include="Properties\Resources.resx">
<Generator>ResXFileCodeGenerator</Generator>
<LastGenOutput>Resources.Designer.cs</LastGenOutput>
<SubType>Designer</SubType>
</EmbeddedResource>
<None Include="Properties\Settings.settings">
<Generator>SettingsSingleFileGenerator</Generator>
<LastGenOutput>Settings.Designer.cs</LastGenOutput>
</None>
<AppDesigner Include="Properties\" />
</ItemGroup>
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
<!-- To modify your build process, add your task inside one of the targets below and uncomment it.
Other similar extension points exist, see Microsoft.Common.targets.
<Target Name="BeforeBuild">
</Target>
<Target Name="AfterBuild">
</Target>
-->
</Project>
Now, the combination of <ApplicationDefinition> element and the <Import> tag ‘Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets"’ will create our application entry point.
My ‘App.xaml’ file
<Application x:Class="HelloWorld.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Startup="Application_Startup"
StartupUri="Window1.xaml">
<Application.Resources>
</Application.Resources>
</Application>
Notice the StartUp tag here. The connection between XAML and the code behind can be found in the file ‘App.g.cs’
/// <summary>
/// InitializeComponent
/// </summary>
[System.Diagnostics.DebuggerNonUserCodeAttribute()]
public void InitializeComponent() {
#line 4 "..\..\App.xaml"
this.Startup += new System.Windows.StartupEventHandler(this.Application_Startup);
#line default
#line hidden
#line 5 "..\..\App.xaml"
this.StartupUri = new System.Uri("Window1.xaml", System.UriKind.Relative);
#line default
#line hidden
}
Now, we are sure that ‘App.g.cs’ has a significant role in bridging the event handlers specified in the XAML file to the code behind files (c# files). Since ‘App.g.cs’ is a partical class, may be visual studio uses it for Intellisense. This is not all ?, Where does the XAML file go after the compilation process ?. Notice that if you navigate to .\obj\debug or .obj\release folders you find a Window1.baml file. Now, as i’ve already said before, BAML is an optimized version of XAML.
I found this explanation on the web [1] , dumping the content here.
What tools are used to generate the .G.CS classes when a XAML project is created? Also is there an API for this so I can do it at runtime?
The answer:
In order to compile xaml files, we have declared a few MsBuild tasks in PresentationBuildTasks.dll. Projects that contain <Page Include="Page1.xaml" /> and that import MicrosoftWinFX.targets will be markup compiled.There are 2 "Targets" defined in that targets file that do the work for markup compilation. MarkupCompilePass1 – creates a .g.cs file for each Page. If the xaml doesn't refer to any classes defined in the assembly being created during the build, a baml file is created. This baml file is a binary (pre-tokenized) version of the hierarchy of objects and property sets from the xaml file. MarkupCompilePass2 – creates all baml files that weren't able to be created in Pass1
The only APIs that WPF exposes to do this markup compilation are those msbuild based tasks in PresentationBuildTasks.dll.MsBuild can be called at runtime (from full trust apps).
Check out [2] for more on BAML and .net assembly. If you are thinking of writing your own UI representation like YAML etc, then you might need the BAML Reader/Writer [3].
Some Links to understand what’s going on beneath the hood.
1. Rrelyes, MS
2. A nice little anatomy of a wpf compilation here
Monday, 15 September 2008
PRISM Guidance
PRISM is the composite application design and development guide from Microsoft Patterns & Practises for helping you build rich flexible and easily maintained WPF, Silverlight and Windows Phone apps.
Using design patterns that embody important architectural design principles, such as separation of concerns and loose coupling, Prism helps you to design and build applications using loosely coupled components that can evolve independently but which can be easily and seamlessly integrated into the overall application.
Check out the guidance on MSDN