Skip to content

Latest commit

 

History

History
70 lines (49 loc) · 4.24 KB

README.md

File metadata and controls

70 lines (49 loc) · 4.24 KB

Xamarin Forms ItemsControl

A simple implementation of the ItemsControl in Xamarin Forms. The ItemsControl takes an items source, creates items based on item template, and layouts them using an instance of a layout control.

A video with this sample: https://www.youtube.com/watch?v=PMacCbmg51U

If you're not familiar with ItemsControl in Windows XAML frameworks, see the MSDN docs: https://msdn.microsoft.com/en-us/library/system.windows.controls.itemscontrol(v=vs.110).aspx

Example

    <local:ItemsControl ItemsSource="{Binding Items}">
        <local:ItemsControl.ItemTemplate>
            <DataTemplate>
                <Label BackgroundColor="Magenta"
                       Text="{Binding }"
                       WidthRequest="80"
                       HeightRequest="50" />
            </DataTemplate>
        </local:ItemsControl.ItemTemplate>
        <local:ItemsControl.ItemsLayout>
            <DataTemplate>
                <ScrollView Orientation="Vertical">
                    <local:WrapLayout IsSquare="True"
                                      Spacing="10"
                                      local:LayoutEx.IsItemsHost="True" />
                </ScrollView>
            </DataTemplate>
        </local:ItemsControl.ItemsLayout>
    </local:ItemsControl>

How it works

It's a simple control with little logic, it has an ItemsSource and ItemTemplate properties to create child items, but it delegates their positioning to a Layout<View> instance. It does not use any renderers.

This allows displaying a bindable items source by any Layout<View> derived class. Xamarin Forms has some built-in layouts like StackLayout, Grid, AbsoluteLayout, RelativeLayout. Or you can have your own layout to position items in a specific way. In this sample app to demo the control I have a WrapLayout.

By default, ItemsControl uses a StackLayout to position the items. You don't need to set the ItemsLayout property if all you need is stacking items vertically.

Being able to easily switch between how items are positioned can be extremely handy when you have a UI like a file explorer for example. The Layout makes it share and reuse layout logic.

Similar to Windows XAML, if the items layout template needs to be a more complicated hierarchy where the actual layout control is not the root, you need to set IsItemsHost property so that ItemsControl knows which is the actual layout where to add the items created by ItemsSource and ItemTemplate.

It's not necessary to to set IsItemsHost=True if the root element of the data template is a Layout<View> derived class:

        <local:ItemsControl.ItemsLayout>
            <DataTemplate>
                 <local:WrapLayout IsSquare="True"
                                   Spacing="10" />
            </DataTemplate>
        </local:ItemsControl.ItemsLayout>
    </local:ItemsControl>

More things I'd like to share

  1. It would make more sense if ItemsControl derived from Xamarin Form's built-in class ItemsView https://developer.xamarin.com/api/type/Xamarin.Forms.ItemsView%3CTVisual%3E/ since ItemsView already has ItemsSource and ItemTemplate properties. I couldn't do this because ItemsView has an internal constructor!

  2. If you run the sample, you will see an issue with the demo main page. On Android, the ItemsControl overflows the lower StackPanel. This looks to be a problem because of Grid which still has issues in Xamarin Forms. On iOS it's worse, the ItemsControl completely takes the whole Grid. I put a BackgroundColor on the StackLayout created implicitly by the ItemsControl https://github.com/andreinitescu/XFItemsControl/blob/master/XFItemsControl/XFItemsControl/ItemsControl.cs#L55 and I could see how the items inside the StackLayout overflow it! I haven't tried it yet on a real iOS device but it doesn't look like a simulator issue.

Is this production ready?

I think so, if you don't need to update the items source :). Current implementation does not check and subscribe to observable collection. The other issue might be with the initial rendering I mentioned in point #2 in previous section. But if you explicitly set a ItemsLayout template, it works.