/Gu.Wpf.AutoGrid

Primary LanguageC#MIT LicenseMIT

Gu.Wpf.AutoGrid

License Build status NuGet

Small prototype for a less noisy WPF grid. Perhaps useful when prototyping.

Grid

Is a MarkupExtension that spits out a new vanilla WPF grid when ProvideValue is called.

ColumnDefinitions

Attribute style

<autoRowGrid:Grid ColumnDefinitions="Auto * 50">
    ...
</autoRowGrid:Grid>

Element style

The designer may show a false negative error: The attachable property 'ColumnDefinitions' was not found in type 'Grid'. Not sure if #pragma can be used in xaml somehow.

<autoRowGrid:Grid AutoIncrementation="UseExplicitColumns">
    <autoRowGrid:Grid.ColumnDefinitions>
        <ColumnDefinition Width="Auto" MinWidth="70" />
        <ColumnDefinition Width="*" MaxWidth="70" />
    </autoRowGrid:Grid.ColumnDefinitions>
	 ...
</autoRowGrid:Grid>

Row

Is just a List<UIElement> that insers its contents on the next row in the grid. This avoids specifying Grid.Row="n"on multiple elements that are oin the same row and makes reordering trivial. Note that Rowonly exists in xaml for grouping elements on the same row. The idea is that it will making rearranging & reasoning about the grid easier. Using a row the need for renumbering Grid.Row when moving things goes away.

AutoIncrementation

By default children of a Row get column index from their position:

<autoRowGrid:Row Name="first row">
    <TextBlock Text="foo1" /> <!--this gets Grid.Column="0"-->
    <TextBox Text="{Binding Value1}" /> <!--this gets Grid.Column="1"-->
</autoRowGrid:Row>

If you want to set expllicit columns use UseExplicitColumns

<autoRowGrid:Row AutoIncrementation="UseExplicitColumns">
    <Rectangle Grid.Column="0" Fill="Aqua" />
    <TextBlock Grid.Column="0" Text="foo1" />
    <TextBox Grid.Column="1" Text="{Binding Value1}" />
</autoRowGrid:Row>

Can also be set on a single UIElement:

<autoRowGrid:Row AutoIncrementation="AutoIncrement">
    <Rectangle Grid.Column="0" Fill="Aqua" autoRowGrid:AutoIncrement.AutoIncrementation="UseExplicitColumns" />
    <TextBlock Text="foo3" />
    <TextBox Text="{Binding Value3}" />
</autoRowGrid:Row>

Setting this on an element excludes it from the count so following elements are not affected.

Defaults

The default for AutoGrid is Increment meaning children for Row get column index from position. The default for Rowand Rows is Inherit meaning they inherit incrementation from parent going all the way up to the GridExtension

LastRowFill

If the last row should fill remaining space of if a starsized row should be added after the last row.

<autoRowGrid:Grid ColumnDefinitions="Auto *" LastRowFill="True">
    <autoRowGrid:Row>
        <TextBlock Text="foo1" />
        <TextBox Text="{Binding Value1}" />
    </autoRowGrid:Row>
    <autoRowGrid:Row AutoIncrementation="UseExplicitColumns">
        <Rectangle Grid.ColumnSpan="2" Fill="RosyBrown" />
        <TextBlock Grid.ColumnSpan="2" HorizontalAlignment="Center" Text="last" />
    </autoRowGrid:Row>
</autoRowGrid:Grid>

RowHeight

Set RowHeight to specify height of generated rows. If set to star LastRowFill will have no effect.

Or specify on individual rows like this:

<autoRowGrid:Grid ColumnDefinitions="* *">
    <autoRowGrid:Row Height="25">
        <Rectangle Fill="Yellow" />
        <Rectangle Fill="Blue" />
    </autoRowGrid:Row>

    <autoRowGrid:Row Height="40">
        <Rectangle Fill="Red" />
        <Rectangle Fill="Yellow" />
    </autoRowGrid:Row>
</autoRowGrid:Grid>

Sample xaml:

<Window ...
        xmlns:autoRowGrid="http://gu.se/AutoRowGrid"
        ...>
    <autoRowGrid:Grid ColumnDefinitions="Auto * 50">
        <TextBlock Grid.Row="0" Grid.Column="0" Text="foo1" />
        <TextBox Grid.Row="0" Grid.Column="1"  Text="{Binding Value1}" />

        <TextBlock Grid.Row="1" Grid.Column="0" Text="foo2" />
        <TextBox Grid.Row="1" Grid.Column="1"  Text="{Binding Value2}" />

        <Rectangle Fill="Yellow" Height="15" Grid.Row="2" Grid.Column="0" Grid.ColumnSpan="3" />

        <autoRowGrid:Row>
            <TextBlock Text="foo3" />
            <TextBox Text="{Binding Value3}" />
        </autoRowGrid:Row>

        <autoRowGrid:Row>
            <TextBlock Text="foo4" />
            <TextBox Text="{Binding Value4}" />
        </autoRowGrid:Row>

        <Button Grid.Row="4" Grid.Column="2" Content="foo5" />

        <autoRowGrid:Row AutoIncrementation="UseExplicitColumns">
            <Rectangle Fill="Blue" Height="15" Grid.Row="2" Grid.Column="1" Grid.ColumnSpan="2" />
        </autoRowGrid:Row>
    </autoRowGrid:Grid>
</Window>

Produces:

screenie

Sample with nesting:

<UserControl ...
             xmlns:autoRowGrid="http://gu.se/AutoRowGrid"
             ...>
    <autoRowGrid:Grid ColumnDefinitions="Auto *">
        <autoRowGrid:Row Name="first row">
            <TextBlock Text="foo1" />
            <TextBox Text="{Binding Value1}" />
        </autoRowGrid:Row>

        <autoRowGrid:Rows Name="a bunch of rows">
            <autoRowGrid:Row>
                <TextBlock Text="foo2" />
                <TextBox Text="{Binding Value2}" />
            </autoRowGrid:Row>

            <autoRowGrid:Row>
                <TextBlock Text="foo3" />
                <TextBox Text="{Binding Value3}" />
            </autoRowGrid:Row>

            <autoRowGrid:Rows Name="a bunch of nested rows">
                <autoRowGrid:Row>
                    <TextBlock Text="foo4" />
                    <TextBox Text="{Binding Value4}" />
                </autoRowGrid:Row>

                <autoRowGrid:Row>
                    <TextBlock Text="foo5" />
                    <TextBox Text="{Binding Value5}" />
                </autoRowGrid:Row>
            </autoRowGrid:Rows>

            <autoRowGrid:Row>
                <TextBlock Text="foo6" />
                <TextBox Text="{Binding Value6}" />
            </autoRowGrid:Row>
        </autoRowGrid:Rows>

        <autoRowGrid:Row Name="last row">
            <TextBlock Text="foo7" />
            <TextBox Text="{Binding Value7}" />
        </autoRowGrid:Row>
    </autoRowGrid:Grid>
</UserControl>

Produces: screenie

Limitation

As it is a markupextension using it like this will not work:

<Grid>
    <Grid.RowDefinitions>
        <RowDefinition />
        <RowDefinition />
    </Grid.RowDefinitions>
	<autoRowGrid:Grid>
		 ...
	</autoRowGrid:Grid>
</Grid>

error

Not sure if there is a nice workaround for this. Think the lib still has potential to be useful. A hacky way would be to set it as content for a lightweight element, probably good with a comment in xaml for stuff like this:

<Grid>
    <Grid.RowDefinitions>
        <RowDefinition />
        <RowDefinition />
    </Grid.RowDefinitions>
    <Border>
        <!--using a border to set the grid as content for here, the border does nothing but is a lightweight element-->
        <autoRowGrid:Grid ColumnDefinitions="*">
            <TextBlock Text="hej" />
        </autoRowGrid:Grid>
    </Border>
</Grid>
´´´