Windows Phone 8的LongListSelector控件按拼音分组主要有两种方法,一个是在数据源里手工指定拼音首字母字段,作为index,这种方法效率高但会造成数据冗余不宜维护。另一个就是我今天介绍的方法,来自MSDN,虽然官网例子是针对是英文数据的首字母分组,但其实稍微改一下还是是支持中文的。

首先给大家看看分组的效果,和WP人脉应用按联系人拼音首字母分组的意思是一样的:

实现方法很简单。首先你需要一个来自MSDN的AlphaKeyGroup类,代码如下(我稍作了代码风格上的改动,编译结果和MSDN官网是一样的,不要在意这些细节):

public class AlphaKeyGroup<T> : List<T>
{
    /// <summary>
    /// The delegate that is used to get the key information.
    /// </summary>
    /// <param name="item">An object of type T</param>
    /// <returns>The key value to use for this object</returns>
    public delegate string GetKeyDelegate(T item);

    /// <summary>
    /// The Key of this group.
    /// </summary>
    public string Key { get; private set; }

    /// <summary>
    /// Public constructor.
    /// </summary>
    /// <param name="key">The key for this group.</param>
    public AlphaKeyGroup(string key)
    {
        Key = key;
    }

    /// <summary>
    /// Create a list of AlphaGroup<T> with keys set by a SortedLocaleGrouping.
    /// </summary>
    /// <param name="slg">The </param>
    /// <returns>Theitems source for a LongListSelector</returns>
    private static List<AlphaKeyGroup<T>> CreateGroups(SortedLocaleGrouping slg)
    {
        return slg.GroupDisplayNames.Select(key => new AlphaKeyGroup<T>(key)).ToList();
    }

    /// <summary>
    /// Create a list of AlphaGroup<T> with keys set by a SortedLocaleGrouping.
    /// </summary>
    /// <param name="items">The items to place in the groups.</param>
    /// <param name="ci">The CultureInfo to group and sort by.</param>
    /// <param name="getKey">A delegate to get the key from an item.</param>
    /// <param name="sort">Will sort the data if true.</param>
    /// <returns>An items source for a LongListSelector</returns>
    public static List<AlphaKeyGroup<T>> CreateGroups(IEnumerable<T> items, CultureInfo ci, GetKeyDelegate getKey, bool sort)
    {
        var slg = new SortedLocaleGrouping(ci);
        var list = CreateGroups(slg);

        foreach (var item in items)
        {
            var index = 0;
            if (slg.SupportsPhonetics)
            {
                //check if your database has yomi string for item
                //if it does not, then do you want to generate Yomi or ask the user for this item.
                //index = slg.GetGroupIndex(getKey(Yomiof(item)));
            }
            else
            {
                index = slg.GetGroupIndex(getKey(item));
            }
            if (index >= 0 && index < list.Count)
            {
                list[index].Add(item);
            }
        }

        if (!sort) return list;
        foreach (var group in list)
        {
            @group.Sort((c0, c1) => ci.CompareInfo.Compare(getKey(c0), getKey(c1)));
        }

        return list;
    }

}

这个类里面主要的精髓就在于:

var slg = new SortedLocaleGrouping(ci);

SortedLocaleGrouping可以根据传入的传入的CultureInfo返回经过排序的组标头。

要按照拼音首字母排序,我们只要传入中国大陆的CultureInfo就可以了,也就是zh-CN。在中文环境的Windows Phone系统上,当然也可以用当前UI线程的CultureInfo去获得

System.Threading.Thread.CurrentThread.CurrentUICulture

但是为了保证我们的拼音排序能在任何语言设置下都统一,我还是建议写死zh-CN在里面。

在我的应用里,我需要按地铁站点(Station类)的地铁站名首字母(Station.StationName)分组,所以我绑定的集合要用AlphaKeyGroup包一下:

public ObservableCollection<AlphaKeyGroup<Station>> GroupedStations { ... }

然后,在给这个集合赋值的地方写Group的具体逻辑:

GroupedStations = AlphaKeyGroup<Station>.CreateGroups(
    AllStations,
    new CultureInfo("zh-CN"),
    s => s.StationName.Substring(0, 1),
    true).ToObservableCollection();

第一个参数AllStations是原始数据,一个普通的IEnumerable<Station>集合。

第二个参数是最重要的,按哪种Culture进行分组,一定要传入zh-CN,简体中文。

第三个参数是个lambda表达式,这个委托负责分组字段的具体逻辑,在这里我们要按Station.StationName的第一个字的拼音首字母排序,所以需要取Substring(0,1),返回Station.StationName的第一个字,之后SortedLocaleGrouping就可以自动进行拼音首字母分组了。

第四个参数表示经过分组的排序结果需不需要排序,true表示需要排序,这也是我们通常的需求。

至此,后端代码的工作就全部搞定了。前台xaml上的数据绑定还是和普通的LongListSelector没啥区别。

XAML:

<phone:LongListSelector
    x:Name="StationListSelector"
    ItemsSource="{Binding GroupedStations, Mode=TwoWay}" 
    JumpListStyle="{StaticResource StationListJumpListStyle}"
    IsGroupingEnabled="True"
    HideEmptyGroups="True"
    SelectionChanged="SelectStation"
    toolkit:TiltEffect.IsTiltEnabled="True">

    <phone:LongListSelector.GroupHeaderTemplate>
        <DataTemplate>
            <Border 
                BorderBrush="{StaticResource PhoneAccentBrush}"
                Padding="15,0"
                Width="75"
                Height="75"
                Margin="0,0,0,10"
                BorderThickness="2"
                HorizontalAlignment="Left" 
                    Background="White">
                <TextBlock Text="{Binding Key}" 
                           FontSize="48"
                           Foreground="{StaticResource PhoneAccentBrush}">
                </TextBlock>
            </Border>
        </DataTemplate>
    </phone:LongListSelector.GroupHeaderTemplate>
....

App.xaml里的Style:

<Style x:Key="StationListJumpListStyle" TargetType="phone:LongListSelector">
    <Setter Property="GridCellSize"  Value="113,113"/>
    <Setter Property="LayoutMode" Value="Grid" />
    <Setter Property="ItemTemplate">
        <Setter.Value>
            <DataTemplate>
                <Border Background="{StaticResource PhoneAccentBrush}" Width="113" Height="113" Margin="6" >
                    <TextBlock Text="{Binding Key}" FontFamily="{StaticResource PhoneFontFamilySemiBold}" FontSize="48" Padding="6"
       Foreground="{StaticResource PhoneForegroundBrush}" VerticalAlignment="Center"/>
                </Border>
            </DataTemplate>
        </Setter.Value>
    </Setter>
</Style>