With the latest update to the Windows Phone Toolkit they have overhauled the internals of the LongListSelector for the Mango release. One of the changes was removing suppor
The LongListSelector
is using a ScrollViewer
internally (apparently since the Aug 2011 release). This fact can be used to restore the position of the list after tombstoning by following the example given at http://damianblog.com/2011/01/21/wp7-scroll-pivot/ for the pivot controller.
In OnNavigatedFrom()
remember the scroll offset:
private bool _newPageInstance = true;
private double _scollOffset = double.NaN;
protected override void OnNavigatedFrom(System.Windows.Navigation.NavigationEventArgs e)
{
base.OnNavigatedFrom(e);
var scrollViewer = FindScrollViewer(LongList);
State["scrollViewer.VerticalOffset"] = scrollViewer.VerticalOffset;
State["PreservingPageState"] = true;
_newPageInstance = false;
}
private static ScrollViewer FindScrollViewer(DependencyObject parent)
{
var childCount = VisualTreeHelper.GetChildrenCount(parent);
for (var i = 0; i < childCount; i++)
{
var elt = VisualTreeHelper.GetChild(parent, i);
if (elt is ScrollViewer) return (ScrollViewer)elt;
var result = FindScrollViewer(elt);
if (result != null) return result;
}
return null;
}
And restore it in OnNavigatedTo()
if the app have been tombstoned:
protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
{
base.OnNavigatedTo(e);
if (!_newPageInstance || !State.ContainsKey("PreservingPageState"))
{
return;
}
_scollOffset = (double)State["scrollViewer.VerticalOffset"];
}
private void LongList_Loaded(object sender, RoutedEventArgs e)
{
if (double.IsNaN(_scollOffset)) return;
var longListSelector = (LongListSelector)sender;
var scrollViewer = FindScrollViewer(longListSelector);
scrollViewer.ScrollToVerticalOffset(_scollOffset);
_scollOffset = double.NaN;
}
From what I can tell from the new bits, you have to subscribe to the LLS's Link
and Unlink
events. Link
will pass in an arg that contains the item added to the visible part of the LLS. Unlink
does the same for those items removed from the LLS. So you'd do something like this:
List<string> trackedItems = new List<string>();
private void myListOfStrings_Link(object sender, LinkUnlinkEventArgs e)
{
var x = e.ContentPresenter;
if (x == null || x.Content == null)
return;
trackedItems.Add(x.Content.ToString());
}
private void myListOfString_Unlink(object sender, LinkUnlinkEventArgs e)
{
var x = e.ContentPresenter;
if (x == null || x.Content == null)
return;
trackedItems.Remove(x.Content.ToString());
}
Note that Link
and Unlink
will fire for EVERY rendered item in the underlying list, so if you're using the grouping features of the LLS, then you'll have to augment your test on whether or not to track the item based on what type is actually coming back. So if you have some sort of group object for which you want to track the underrlying objects, you might do something like this:
private void myGroupedListOfObjects_Link(object sender, LinkUnlinkEventArgs e)
{
var x = e.ContentPresenter;
if (x == null || x.Content == null)
return;
var myObject = x.Content as MyObject;
if (myObject != null)
{
foreach (var item in myObject.Items)
{
trackedItems.Add(item);
}
}
}
I hope this helps! Let us know if it works out.
The Link/Unlink approach does not work at all for restoring the scroll position. Even if you setup the collection, you don't know if you are scrolling up or down, and the size of the collection would vary depending on the BufferSize
property of the LongListSelector
.
The FindScrollViewer
solution in kvakulo's answer, however, works.
If someone needs the VB.Net version of this code:
Friend Function FindScrollViewer(parent As DependencyObject) As ScrollViewer
Dim childCount = VisualTreeHelper.GetChildrenCount(parent)
For i As Int32 = 0 To childCount - 1
Dim elt = VisualTreeHelper.GetChild(parent, i)
If elt.GetType Is GetType(ScrollViewer) Then Return CType(elt, ScrollViewer)
Dim result = FindScrollViewer(elt)
If result IsNot Nothing Then Return result
Next
Return Nothing
End Function