This Xamarin Workshop Guide was created for the The Portuguese National Meeting of IT Students (ENEI)by Sara Silva of which the original content is available here. To extend it to the global community, it was published in a new project called Xam Community Workshop that the main goal is for any developer or user group to customize it to their events.
Before reading this article you must read:
- Xamarin Guide 1: Create a Xamarin Forms Project
- Xamarin Guide 2: Create the Model and the Data Source
In this step, you will create the user interface to the SessionsView (which is the main page) to show the data defined in the last step.
3.1. Create the XAML page
In Xamarin Studio, select the ENEI.SessionsApp project and create a folder called “Views”, as described in Figure 1 and Figure 2.
Figure 1: Creating new folder
Figure 2: The Views folder in the project
Then select the folder “Views” then double-click the mouse to open the context and the click on “Add” > “New File...” as shown in Figure 3:
Figure 3: Add new file
Select “Forms ContentPage XAML”, as in the following:
Figure 4: Adding a XAML page
The result will be something as in the following:
Figure 5: The XAML page code
The SessionsView XAML is defined by:
- The XAML file that defines the user interface
- The cs file that defines the code behind
This approach is really useful when designers and developers work together and using the MVVM pattern allows you to have a UI separated from the behavior of the page.
See more about Xamarin.Forms XAML Basics.
See more about Xamarin.Forms XAML Basics.
Now let's change the App.cs in ENEI.SessionsApp to use the SessionsView:
In this case, you defined the MainPage with a NavigationPage that the content is defined by SessionView. The NavigationPage is required when the application requires navigation between a page and the application that should have only one NavigationPage.
At this moment, if you run the application you will not have content and you should have something such as described in Figure 6.
Figure 6: The XAML page code
Now let's define the UI.
3.2. Create the Header
The first thing you should define is the Title and the Icon used in the header of the page, something such as in the following:
Figure 7: The header
That code can be:
This will be showed in Android and in iOS but for Windows Phone 8.0 we need to create a workaround for it, as we will see below.
Figure 8: Control Layouts
This way, let's choose the Grid as control layout to define two rows and one columns, where:
- public class App : Application
- {
- public App()
- {
- // The root page of your application
- MainPage = new NavigationPage(new SessionsView())
- {
- BarBackgroundColor = Color.White,
- BarTextColor = Color.Black
- };
- }
- }
At this moment, if you run the application you will not have content and you should have something such as described in Figure 6.
Figure 6: The XAML page code
Now let's define the UI.
3.2. Create the Header
The first thing you should define is the Title and the Icon used in the header of the page, something such as in the following:
Figure 7: The header
That code can be:
- <ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
- xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
- x:Class="ENEI.SessionsApp.Views.SessionsView"
- Title="1010 ENEI || Sessões"
- BackgroundColor="White"
- Icon="ic_action_users.png">
In Android, more specific in the MainActivity is required (in this case) to set the Icon in ActionBar as in the following: ActionBar.SetIcon(Resource.Drawable.ic_action_users);
The SessionsView is a ContentPage that is a simple page provided by Xamarin Forms API (see more inXamarin Forms Gallery). To define its content, you should use controls layouts like StackLayout or Grid, for example.Figure 8: Control Layouts
This way, let's choose the Grid as control layout to define two rows and one columns, where:
- The first row has the header for Windows Phone (hidden to the others platforms)
- The second row has the ListView to show the sessions
The code for it will be something such as in the following:
- <?xml version="1.0" encoding="utf-8" ?>
- <ContentPage xmlns="http://xamarin.com/schemas/2014/forms"
- xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
- x:Class="ENEI.SessionsApp.Views.SessionsView"
- Title="1010 ENEI || Sessões"
- BackgroundColor="White"
- Icon="ic_action_users.png">
- <Grid BackgroundColor="White">
- <Grid.RowDefinitions>
- <RowDefinition Height="Auto" />
- <RowDefinition Height="*" />
- </Grid.RowDefinitions>
- <!-- Title - Only for WP-->
- <StackLayout Grid.Row="0" Orientation="Horizontal" Padding="20,10,0,0">
- <StackLayout.IsVisible>
- <OnPlatform Android="false"
- WinPhone="true"
- iOS="false"
- x:TypeArguments="x:Boolean" />
- </StackLayout.IsVisible>
- <Image WidthRequest="48"
- HeightRequest="38"
- Source="Images/ic_action_users.png"/>
- <Label FontSize="Large" FontAttributes="Bold"
- TextColor="Black">
- <OnPlatform Android=""
- WinPhone="1010 ENEI || Sessões"
- iOS=""
- x:TypeArguments="x:String" />
- </Label>
- </StackLayout>
- <!-- ListView will be defined here -->
- </Grid>
- </ContentPage>
In the last step you used the image “ic_action_users.png” that must be added to each project (ENEI.SessionApp.Android, ENEI.SessionApp.iOS and ENEI.SessionApp.WinPhone). This way, each app will have its own images that should be defined following the platform specifications, in other words, each image should provide the right resolution by platform.
Get the images and assets by platform here and see more about this subject in this article.
At this moment, you should have:
Figure 9: The Windows Phone, iOS and Android applications
3.4 Creating the Listview
The next step is to define the list of the sessions from the 1010 ENEI that were defined in the topic “The Data Source”.
To show the list of the sessions you will use a ListView that must have:
- ItemsSource defined with the list of sessions
- ItemTemplate defined with the template for each row
Let's define the first version of the ListView as in the following:
- <!-- ListView will be defined here -->
- <ListView x:Name="SessionsList"
- Grid.Row="1"
- ItemSelected="SessionsList_OnItemSelected"
- ItemsSource="{Binding Sessions}"
- SeparatorColor="#0094FF">
- <!--
- Setting the HasUnevenRows property tells the list view to render
- each cell with a different height.
- -->
- <ListView.RowHeight>
- <OnPlatform Android="150"
- WinPhone="180"
- iOS="150"
- x:TypeArguments="x:Int32" />
- </ListView.RowHeight>
- </ListView>
Data bindings allow properties of two objects to be linked so that a change in one causes a change in the other. See more about it in these article Data Binding Basics and From Data Bindings to MVVM
Each row from the ListView can be defined with a static or dynamic size. In this case a static size was defined for each platform (related with screen resolution). A developer that needs to have a different row's size depending on the data shown is recommended to use the property HasUnevenRows.To complete, in the code behind, you need to define the Sessions that should be an “ObservableCollection” of Session (this kind of list is allowed to notify the UI each time an object is removed or added in the list). You need to get the sessions from the “SessionsDataSource” and you need to define the “BindingContext”. The data should be loaded in the “OnAppearing” method and is not recommended to load it in the constructor of the page, because it will increase the time required to create the page and that can create issues.
- public partial class SessionsView : ContentPage
- {
- public SessionsView()
- {
- InitializeComponent();
- Sessions = new ObservableCollection<Session>();
- BindingContext = this;
- }
- public ObservableCollection<Session> Sessions { get; set; }
- protected override void OnAppearing()
- {
- base.OnAppearing();
- if (Sessions.Count == 0)
- {
- var sessions = SessionsDataSource.GetSessions();
- foreach (var session in sessions)
- {
- Sessions.Add(session);
- }
- }
- }
- private void SessionsList_OnItemSelected(object sender, SelectedItemChangedEventArgs e)
- {
- if (SessionsList.SelectedItem == null)
- {
- return;
- }
- SessionsList.SelectedItem = null;
- }
- }
In this case, you will not use the MVVM pattern that is extremely recommended in real projects. To keep it simple the code behind will have the behavior used by the UI that in a MVVM pattern is defined in the ViewModel.
3.5 Creating the ItemTemplate
The ListView has the “ItemTemplate” property that allows defining a “DataTemplate” for each row. In this case you will define a template as described in Figure 10, that code will be:
The ListView has the “ItemTemplate” property that allows defining a “DataTemplate” for each row. In this case you will define a template as described in Figure 10, that code will be:
- <ListView.ItemTemplate>
- <DataTemplate>
- <ViewCell>
- <ViewCell.View>
- <Grid BackgroundColor="Transparent" Padding="20,0,20,0">
- <Grid.ColumnDefinitions>
- <ColumnDefinition Width="Auto" />
- <ColumnDefinition Width="*" />
- </Grid.ColumnDefinitions>
- <Grid.RowDefinitions>
- <RowDefinition Height="5" />
- <RowDefinition Height="Auto" />
- <RowDefinition Height="48" />
- <RowDefinition Height="5" />
- </Grid.RowDefinitions>
- <!-- Define the Image from Speaker -->
- <Image Grid.Row="1"
- HorizontalOptions="StartAndExpand"
- Source="{Binding Speaker.ImageUrl}"
- VerticalOptions="Start">
- <Image.WidthRequest>
- <OnPlatform Android="50"
- WinPhone="100" iOS="50"
- x:TypeArguments="x:Double" />
- </Image.WidthRequest>
- <Image.HeightRequest>
- <OnPlatform Android="50"
- WinPhone="100" iOS="50"
- x:TypeArguments="x:Double" />
- </Image.HeightRequest>
- </Image>
- <!-- Define the Image from Speaker -->
- <StackLayout Grid.Row="1"
- Grid.Column="1"
- HorizontalOptions="FillAndExpand"
- Padding="10,0,0,0">
- <Label Font="Medium"
- FontAttributes="Bold"
- Text="{Binding Name}"
- TextColor="Black" />
- <Label Font="Medium"
- LineBreakMode="TailTruncation"
- Text="{Binding Speaker.Name}"
- TextColor="Black" />
- <Label Font="Small"
- LineBreakMode="TailTruncation"
- TextColor="Black" Text="{Binding Details}"/>
- </StackLayout>
- <!-- Define the menu for each session -->
- </Grid>
- </ViewCell.View>
- </ViewCell>
- </DataTemplate>
- </ListView.ItemTemplate>
Figure 10: A row in ListView
Running the application you will have the following:
Figure 11: The Windows Phone, iOS and Android application showing sessions from 1010 ENEI
3.6. Create the menu for each Session
To create the menu for each session as described in Figure 12, you need to change the ItemTemplate defined above:
Figure 12: The session's menu
- <Grid Grid.Row="2" Grid.Column="0" Grid.ColumnSpan="2" Padding="0,5,0,0">
- <Grid.ColumnDefinitions>
- <ColumnDefinition Width="*" /> <ColumnDefinition Width="Auto" />
- <ColumnDefinition Width="*" /> <ColumnDefinition Width="Auto" /><ColumnDefinition Width="*" />
- <ColumnDefinition Width="Auto" /> <ColumnDefinition Width="*" /><ColumnDefinition Width="Auto" />
- <ColumnDefinition Width="*" />
- </Grid.ColumnDefinitions>
- <StackLayout Grid.Column="1" Orientation="Horizontal">
- <Image>
- <Image.WidthRequest>
- <OnPlatform Android="48" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.WidthRequest>
- <Image.HeightRequest>
- <OnPlatform Android="48" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.HeightRequest>
- <Image.Source>
- <OnPlatform x:TypeArguments="ImageSource">
- <OnPlatform.iOS> <FileImageSource File="ic_action_like.png" />
- </OnPlatform.iOS>
- <OnPlatform.Android> <FileImageSource File="ic_action_like.png" />
- </OnPlatform.Android>
- <OnPlatform.WinPhone><FileImageSource File="Images/ic_action_like.png" />
- </OnPlatform.WinPhone>
- </OnPlatform>
- </Image.Source>
- </Image>
- <Label Font="Small" Text="{Binding NumLikes}" TextColor="#0094FF" VerticalOptions="Center" />
- </StackLayout>
- <Image Grid.Column="3" Source="{Binding IsFavorite, Converter={StaticResource FavoriteImageConverter}}">
- <Image.WidthRequest>
- <OnPlatform Android="48" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.WidthRequest>
- <Image.HeightRequest>
- <OnPlatform Android="48" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.HeightRequest>
- </Image>
- <Image Grid.Column="5">
- <Image.WidthRequest>
- <OnPlatform Android="30" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.WidthRequest>
- <Image.HeightRequest>
- <OnPlatform Android="30" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.HeightRequest>
- <Image.Source>
- <OnPlatform x:TypeArguments="ImageSource">
- <OnPlatform.iOS><FileImageSource File="ic_action_share_2.png" /></OnPlatform.iOS>
- <OnPlatform.Android><FileImageSource File="ic_action_share_2.png" /></OnPlatform.Android>
- <OnPlatform.WinPhone>
- <FileImageSource File="Images/ic_action_share_2.png" /></OnPlatform.WinPhone>
- </OnPlatform> </Image.Source> </Image>
- <Image Grid.Column="7">
- <Image.WidthRequest>
- <OnPlatform Android="30" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.WidthRequest>
- <Image.HeightRequest>
- <OnPlatform Android="30" WinPhone="48" iOS="30" x:TypeArguments="x:Double" />
- </Image.HeightRequest>
- <Image.Source>
- <OnPlatform x:TypeArguments="ImageSource">
- <OnPlatform.iOS> <FileImageSource File="ic_action_list.png" />
- </OnPlatform.iOS>
- <OnPlatform.Android><FileImageSource File="ic_action_list.png" />
- </OnPlatform.Android>
- <OnPlatform.WinPhone> <FileImageSource File="Images/ic_action_list.png" />
- </OnPlatform.WinPhone>
- </OnPlatform>
- </Image.Source>
- </Image>
- </Grid>
The code above contains a grid with the four images such as Favorites, Shares and Details. And for each one we need to subscribe to the Tap event using the “GestureRecognizers” from the Image, this way you need to define for each image its TapGesture as in the following:
- <Image.GestureRecognizers>
- <TapGestureRecognizer x:Name="FavoriteGesture"
- CommandParameter="{Binding}" Tapped="FavoriteGestureRecognizer_OnTapped" />
- </Image.GestureRecognizers>
Where the event's handler will be defined as in the following:
- private void FavoriteGestureRecognizer_OnTapped(object sender, EventArgs e)
- {
- var tappedEventArg = e as TappedEventArgs;
- if (tappedEventArg != null)
- {
- var session = tappedEventArg.Parameter as Session;
- if (session != null)
- {
- session.IsFavorite = !session.IsFavorite;
- }
- }
- }
The Favorite option uses a FavoriteImageConverter that allows the showiing of the right image based on if the user selected it as a favorite or not, this way if the favorite option is Red then that means the user selected the session as a favorite, the Blue color is defined by default.
The implementation of the FavoriteImageConverter will be something as in the following:
- public class FavoriteImageConverter:IValueConverter
- {
- public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
- {
- if (value == null)
- {
- return Device.OS == TargetPlatform.WinPhone ? "Images/ic_action_heart.png" : "ic_action_heart.png";
- }
- var isFavorite = (bool) value;
- if (isFavorite)
- {
- return Device.OS == TargetPlatform.WinPhone ? "Images/ic_action_heart_red.png" : "ic_action_heart_red.png";
- }
- return Device.OS == TargetPlatform.WinPhone ? "Images/ic_action_heart.png" : "ic_action_heart.png";
- }
- public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
- {
- throw new NotImplementedException();
- }
- }
At this moment you should have the UI from the 1010 ENEI Sessions App as described in Figure 13:
Figure 13: The Windows Phone, iOS and Android applications
Comments
Post a Comment