316 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			316 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //*********************************************************
 | |
| //
 | |
| // Copyright (c) Microsoft. All rights reserved.
 | |
| // THIS CODE IS PROVIDED *AS IS* WITHOUT WARRANTY OF
 | |
| // ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING ANY
 | |
| // IMPLIED WARRANTIES OF FITNESS FOR A PARTICULAR
 | |
| // PURPOSE, MERCHANTABILITY, OR NON-INFRINGEMENT.
 | |
| //
 | |
| //*********************************************************
 | |
| 
 | |
| //
 | |
| // MainPage.xaml.cpp
 | |
| // Implementation of the MainPage.xaml class.
 | |
| //
 | |
| 
 | |
| #include "pch.h"
 | |
| #include "MainPage.xaml.h"
 | |
| #include "App.xaml.h"
 | |
| 
 | |
| #include <collection.h>
 | |
| 
 | |
| using namespace Windows::UI::Xaml;
 | |
| using namespace Windows::UI::Xaml::Controls;
 | |
| using namespace Windows::Foundation;
 | |
| using namespace Windows::Foundation::Collections;
 | |
| using namespace Platform;
 | |
| using namespace SDKSample;
 | |
| using namespace Windows::UI::Xaml::Navigation;
 | |
| using namespace Windows::UI::Xaml::Interop;
 | |
| using namespace Windows::Graphics::Display;
 | |
| using namespace Windows::UI::ViewManagement;
 | |
| 
 | |
| MainPage^ MainPage::Current = nullptr;
 | |
| 
 | |
| MainPage::MainPage()
 | |
| {
 | |
|     InitializeComponent();
 | |
| 
 | |
|     // This frame is hidden, meaning it is never shown.  It is simply used to load
 | |
|     // each scenario page and then pluck out the input and output sections and
 | |
|     // place them into the UserControls on the main page.
 | |
|     HiddenFrame = ref new Windows::UI::Xaml::Controls::Frame();
 | |
|     HiddenFrame->Visibility = Windows::UI::Xaml::Visibility::Collapsed;
 | |
|     ContentRoot->Children->Append(HiddenFrame);
 | |
| 
 | |
|     FeatureName->Text = FEATURE_NAME;
 | |
| 
 | |
|     this->SizeChanged += ref new SizeChangedEventHandler(this, &MainPage::MainPage_SizeChanged);
 | |
|     Scenarios->SelectionChanged += ref new SelectionChangedEventHandler(this, &MainPage::Scenarios_SelectionChanged);
 | |
| 
 | |
|     MainPage::Current = this;
 | |
|     autoSizeInputSectionWhenSnapped = true;
 | |
| }
 | |
| 
 | |
| /// <summary>
 | |
| /// We need to handle SizeChanged so that we can make the sample layout property
 | |
| /// in the various layouts.
 | |
| /// </summary>
 | |
| /// <param name="sender"></param>
 | |
| /// <param name="e"></param>
 | |
| void MainPage::MainPage_SizeChanged(Object^ sender, SizeChangedEventArgs^ e)
 | |
| {
 | |
|     InvalidateSize();
 | |
|     MainPageSizeChangedEventArgs^ args = ref new MainPageSizeChangedEventArgs();
 | |
|     args->ViewState = ApplicationView::Value;
 | |
|     MainPageResized(this, args);
 | |
| 
 | |
| }
 | |
| 
 | |
| void MainPage::InvalidateSize()
 | |
| {
 | |
|     // Get the window width
 | |
|     double windowWidth = this->ActualWidth;
 | |
| 
 | |
|     if (windowWidth != 0.0)
 | |
|     {
 | |
|         // Get the width of the ListBox.
 | |
|         double listBoxWidth = Scenarios->ActualWidth;
 | |
| 
 | |
|         // Is the ListBox using any margins that we need to consider?
 | |
|         double listBoxMarginLeft = Scenarios->Margin.Left;
 | |
|         double listBoxMarginRight = Scenarios->Margin.Right;
 | |
| 
 | |
|         // Figure out how much room is left after considering the list box width
 | |
|         double availableWidth = windowWidth - listBoxWidth;
 | |
| 
 | |
|         // Is the top most child using margins?
 | |
|         double layoutRootMarginLeft = ContentRoot->Margin.Left;
 | |
|         double layoutRootMarginRight = ContentRoot->Margin.Right;
 | |
| 
 | |
|         // We have different widths to use depending on the view state
 | |
|         if (ApplicationView::Value != ApplicationViewState::Snapped)
 | |
|         {
 | |
|             // Make us as big as the the left over space, factoring in the ListBox width, the ListBox margins.
 | |
|             // and the LayoutRoot's margins
 | |
|             InputSection->Width = ((availableWidth) -
 | |
|                 (layoutRootMarginLeft + layoutRootMarginRight + listBoxMarginLeft + listBoxMarginRight));
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             // Make us as big as the left over space, factoring in just the LayoutRoot's margins.
 | |
|             if (autoSizeInputSectionWhenSnapped)
 | |
|             {
 | |
|                 InputSection->Width = (windowWidth - (layoutRootMarginLeft + layoutRootMarginRight));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     InvalidateViewState();
 | |
| }
 | |
| 
 | |
| void MainPage::InvalidateViewState()
 | |
| {
 | |
|     // Are we going to snapped mode?
 | |
|     if (ApplicationView::Value == ApplicationViewState::Snapped)
 | |
|     {
 | |
|         Grid::SetRow(DescriptionText, 3);
 | |
|         Grid::SetColumn(DescriptionText, 0);
 | |
| 
 | |
|         Grid::SetRow(InputSection, 4);
 | |
|         Grid::SetColumn(InputSection, 0);
 | |
| 
 | |
|         Grid::SetRow(FooterPanel, 2);
 | |
|         Grid::SetColumn(FooterPanel, 0);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         Grid::SetRow(DescriptionText, 1);
 | |
|         Grid::SetColumn(DescriptionText, 1);
 | |
| 
 | |
|         Grid::SetRow(InputSection, 2);
 | |
|         Grid::SetColumn(InputSection, 1);
 | |
| 
 | |
|         Grid::SetRow(FooterPanel, 1);
 | |
|         Grid::SetColumn(FooterPanel, 1);
 | |
|     }
 | |
| 
 | |
|     //  Since we don't load the scenario page in the traditional manner (we just pluck out the
 | |
|     // input and output sections from the page) we need to ensure that any VSM code used
 | |
|     // by the scenario's input and output sections is fired.
 | |
|     VisualStateManager::GoToState(InputSection, "Input" + LayoutAwarePage::DetermineVisualState(ApplicationView::Value), false);
 | |
|     VisualStateManager::GoToState(OutputSection, "Output" + LayoutAwarePage::DetermineVisualState(ApplicationView::Value), false);
 | |
| }
 | |
| 
 | |
| void MainPage::PopulateScenarios()
 | |
| {
 | |
|     ScenarioList = ref new Platform::Collections::Vector<Object^>();
 | |
| 
 | |
|     // Populate the ListBox with the list of scenarios as defined in Constants.cpp.
 | |
|     for (unsigned int i = 0; i < scenarios->Length; ++i)
 | |
|     {
 | |
|         Scenario s = scenarios[i];
 | |
|         ListBoxItem^ item = ref new ListBoxItem();
 | |
|         item->Name = s.ClassName;
 | |
|         item->Content = (i + 1).ToString() + ") " + s.Title;
 | |
|         ScenarioList->Append(item);
 | |
|     }
 | |
| 
 | |
|     // Bind the ListBox to the scenario list.
 | |
|     Scenarios->ItemsSource = ScenarioList;
 | |
|     Scenarios->ScrollIntoView(Scenarios->SelectedItem);
 | |
| }
 | |
| 
 | |
| /// <summary>
 | |
| /// This method is responsible for loading the individual input and output sections for each scenario.  This
 | |
| /// is based on navigating a hidden Frame to the ScenarioX.xaml page and then extracting out the input
 | |
| /// and output sections into the respective UserControl on the main page.
 | |
| /// </summary>
 | |
| /// <param name="scenarioName"></param>
 | |
| void MainPage::LoadScenario(String^ scenarioName)
 | |
| {
 | |
|     autoSizeInputSectionWhenSnapped = true;
 | |
| 
 | |
|     // Load the ScenarioX.xaml file into the Frame.
 | |
|     TypeName scenarioType = {scenarioName, TypeKind::Custom};
 | |
|     HiddenFrame->Navigate(scenarioType, this);
 | |
| 
 | |
|     // Get the top element, the Page, so we can look up the elements
 | |
|     // that represent the input and output sections of the ScenarioX file.
 | |
|     Page^ hiddenPage = safe_cast<Page^>(HiddenFrame->Content);
 | |
| 
 | |
|     // Get each element.
 | |
|     UIElement^ input = safe_cast<UIElement^>(hiddenPage->FindName("Input"));
 | |
|     UIElement^ output = safe_cast<UIElement^>(hiddenPage->FindName("Output"));
 | |
| 
 | |
|     if (input == nullptr)
 | |
|     {
 | |
|         // Malformed input section.
 | |
|         NotifyUser("Cannot load scenario input section for " + scenarioName +
 | |
|             "  Make sure root of input section markup has x:Name of 'Input'", NotifyType::ErrorMessage);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (output == nullptr)
 | |
|     {
 | |
|         // Malformed output section.
 | |
|         NotifyUser("Cannot load scenario output section for " + scenarioName +
 | |
|             "  Make sure root of output section markup has x:Name of 'Output'", NotifyType::ErrorMessage);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // Find the LayoutRoot which parents the input and output sections in the main page.
 | |
|     Panel^ panel = safe_cast<Panel^>(hiddenPage->FindName("LayoutRoot"));
 | |
| 
 | |
|     if (panel != nullptr)
 | |
|     {
 | |
|         unsigned int index = 0;
 | |
|         UIElementCollection^ collection = panel->Children;
 | |
| 
 | |
|         // Get rid of the content that is currently in the intput and output sections.
 | |
|         collection->IndexOf(input, &index);
 | |
|         collection->RemoveAt(index);
 | |
| 
 | |
|         collection->IndexOf(output, &index);
 | |
|         collection->RemoveAt(index);
 | |
| 
 | |
|         // Populate the input and output sections with the newly loaded content.
 | |
|         InputSection->Content = input;
 | |
|         OutputSection->Content = output;
 | |
| 
 | |
|         ScenarioLoaded(this, nullptr);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // Malformed Scenario file.
 | |
|         NotifyUser("Cannot load scenario: " + scenarioName + ".  Make sure root tag in the '" +
 | |
|             scenarioName + "' file has an x:Name of 'LayoutRoot'", NotifyType::ErrorMessage);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainPage::Scenarios_SelectionChanged(Object^ sender, SelectionChangedEventArgs^ e)
 | |
| {
 | |
|     if (Scenarios->SelectedItem != nullptr)
 | |
|     {
 | |
|         NotifyUser("", NotifyType::StatusMessage);
 | |
| 
 | |
|         LoadScenario((safe_cast<ListBoxItem^>(Scenarios->SelectedItem))->Name);
 | |
|         InvalidateSize();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainPage::NotifyUser(String^ strMessage, NotifyType type)
 | |
| {
 | |
|     switch (type)
 | |
|     {
 | |
|     case NotifyType::StatusMessage:
 | |
|         // Use the status message style.
 | |
|         StatusBlock->Style = safe_cast<Windows::UI::Xaml::Style^>(this->Resources->Lookup("StatusStyle"));
 | |
|         break;
 | |
|     case NotifyType::ErrorMessage:
 | |
|         // Use the error message style.
 | |
|         StatusBlock->Style = safe_cast<Windows::UI::Xaml::Style^>(this->Resources->Lookup("ErrorStyle"));
 | |
|         break;
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
|     StatusBlock->Text = strMessage;
 | |
| 
 | |
|     // Collapsed the StatusBlock if it has no text to conserve real estate.
 | |
|     if (StatusBlock->Text != "")
 | |
|     {
 | |
|         StatusBlock->Visibility = Windows::UI::Xaml::Visibility::Visible;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         StatusBlock->Visibility = Windows::UI::Xaml::Visibility::Collapsed;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainPage::Footer_Click(Object^ sender, RoutedEventArgs^ e)
 | |
| {
 | |
|     auto uri = ref new Uri((String^)((HyperlinkButton^)sender)->Tag);
 | |
|     Windows::System::Launcher::LaunchUriAsync(uri);
 | |
| }
 | |
| 
 | |
| 
 | |
| /// <summary>
 | |
| /// Populates the page with content passed during navigation.  Any saved state is also
 | |
| /// provided when recreating a page from a prior session.
 | |
| /// </summary>
 | |
| /// <param name="navigationParameter">The parameter value passed to
 | |
| /// <see cref="Frame::Navigate(Type, Object)"/> when this page was initially requested.
 | |
| /// </param>
 | |
| /// <param name="pageState">A map of state preserved by this page during an earlier
 | |
| /// session.  This will be null the first time a page is visited.</param>
 | |
| void MainPage::LoadState(Object^ navigationParameter, IMap<String^, Object^>^ pageState)
 | |
| {
 | |
|     (void) navigationParameter;    // Unused parameter
 | |
| 
 | |
|     PopulateScenarios();
 | |
| 
 | |
|     // Starting scenario is the first or based upon a previous state.
 | |
|     ListBoxItem^ startingScenario = nullptr;
 | |
|     int startingScenarioIndex = -1;
 | |
| 
 | |
|     if (pageState != nullptr && pageState->HasKey("SelectedScenarioIndex"))
 | |
|     {
 | |
|         startingScenarioIndex = safe_cast<int>(pageState->Lookup("SelectedScenarioIndex"));
 | |
|     }
 | |
| 
 | |
|     Scenarios->SelectedIndex = startingScenarioIndex != -1 ? startingScenarioIndex : 0;
 | |
| 
 | |
|     InvalidateViewState();
 | |
| }
 | |
| 
 | |
| /// <summary>
 | |
| /// Preserves state associated with this page in case the application is suspended or the
 | |
| /// page is discarded from the navigation cache.  Values must conform to the serialization
 | |
| /// requirements of <see cref="SuspensionManager::SessionState"/>.
 | |
| /// </summary>
 | |
| /// <param name="pageState">An empty map to be populated with serializable state.</param>
 | |
| void MainPage::SaveState(IMap<String^, Object^>^ pageState)
 | |
| {
 | |
|     int selectedListBoxItemIndex = Scenarios->SelectedIndex;
 | |
|     pageState->Insert("SelectedScenarioIndex", selectedListBoxItemIndex);
 | |
| }
 | 
