/* Copyright (c) Citrix Systems, Inc. * All rights reserved. * * Redistribution and use in source and binary forms, * with or without modification, are permitted provided * that the following conditions are met: * * * Redistributions of source code must retain the above * copyright notice, this list of conditions and the * following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the * following disclaimer in the documentation and/or other * materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ using System; using System.Collections.Generic; using System.ComponentModel; using System.Windows.Forms; using XenAdmin.Controls; using XenAdmin.Core; using XenAdmin.Help; using XenAdmin.Network; namespace XenAdmin.Wizards { public partial class XenWizardBase : Form, IFormWithHelp { protected static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType); private IXenConnection connection; // connection to use protected IXenConnection xenConnection { get { return connection; } set { connection = value; foreach (XenTabPage p in wizardProgress.Steps) { p.Connection = connection; } } } protected XenWizardBase() { InitializeComponent(); Icon = Properties.Resources.AppIcon; } protected XenWizardBase(IXenConnection connection) : this() { this.connection = connection; } protected DeprecationBanner Banner { get { return deprecationBanner; } } /// /// Add a given page to the end of the wizard's page collection /// /// The page to add protected void AddPage(XenTabPage page) { AddPage(page, -1); } /// /// Insert a page into the wizard's page collection at the specified index /// /// The page to add /// The index at which the page will be inserted protected void AddPage(XenTabPage page, int index) { xenTabControlBody.Controls.Add(page); page.Dock = DockStyle.Fill; page.WizardContentUpdater = UpdateWizardContent; page.NextPagePrecheck = RunNextPagePrecheck; page.Connection = connection; page.StatusChanged += page_StatusChanged; if (index == -1) wizardProgress.Steps.Add(page); else wizardProgress.Steps.Insert(index, page); } /// /// Add the given pages to the wizard one position after the supplied page /// /// The existing page that the new page will be placed after /// The pages to add protected void AddAfterPage(XenTabPage existingPage, params XenTabPage[] pages) { int index = wizardProgress.Steps.IndexOf(existingPage); for (int i = 0; i < pages.Length; i++) { var page = pages[i]; AddPage(page, index + 1 + i); } } protected void AddPages(params XenTabPage[] pages) { foreach (var page in pages) AddPage(page); } protected void RemovePage(XenTabPage page) { xenTabControlBody.Controls.Remove(page); page.StatusChanged -= page_StatusChanged; page.WizardContentUpdater = null; page.NextPagePrecheck = null; wizardProgress.Steps.Remove(page); } protected void RemovePages(params XenTabPage[] pages) { foreach (var page in pages) RemovePage(page); } /// /// Removes pages at a certain index /// protected void RemovePageAt(int index) { if (wizardProgress.Steps.Count > index) wizardProgress.Steps.RemoveAt(index); } /// /// Removes pages from a certain index and above /// protected void RemovePagesFrom(int startIndex) { if (wizardProgress.Steps.Count > startIndex) wizardProgress.Steps.RemoveRange(startIndex, wizardProgress.Steps.Count - startIndex); } private void WizardProgress_EnteringStep(object sender, WizardProgressEventArgs e) { xenTabControlBody.SelectedTab = wizardProgress.CurrentStepTabPage; wizardProgress.CurrentStepTabPage.PageLoaded(e.IsForwardsTransition ? PageLoadedDirection.Forward : PageLoadedDirection.Back); wizardProgress.CurrentStepTabPage.SelectDefaultControl(); UpdateWizard(); if (wizardProgress.IsLastStep) { // Ensure the finish button always has focus when last page is showing. if (buttonNext.CanFocus) buttonNext.Focus(); } } private void WizardProgress_LeavingStep(object sender, WizardProgressEventArgs e) { wizardProgress.CurrentStepTabPage.PageLeave(e.IsForwardsTransition ? PageLoadedDirection.Forward : PageLoadedDirection.Back, ref e.Cancelled); UpdateWizard(); } #region Button event handlers private void buttonNext_Click(object sender, EventArgs e) { if (wizardProgress.IsLastStep) { var args = new WizardProgressEventArgs(true); WizardProgress_LeavingStep(CurrentStepTabPage, args); if (args.Cancelled) return; wizardFinished = true; WizardProgress_EnteringStep(null, new WizardProgressEventArgs(true)); FinishWizard(); } else { NextStep(); // update main text from selected tab in case it was changed between substeps SetTitle(); } } private void buttonPrevious_Click(object sender, EventArgs e) { wizardProgress.PreviousStep(); // update main text from selected tab in case it was changed between substeps SetTitle(); } private void buttonCancel_Click(object sender, EventArgs e) { Close(); } #endregion /// /// Updates the next wizard pages with content provided by the last visited page /// /// The last visited page protected virtual void UpdateWizardContent(XenTabPage senderPage) { } protected virtual bool RunNextPagePrecheck(XenTabPage senderPage) { return true; } protected void NotifyNextPagesOfChange(params XenTabPage[] pages) { foreach (var p in pages) p.IsFirstLoad = true; } protected virtual void OnCancel(ref bool cancel) { wizardProgress.CurrentStepTabPage.PageCancelled(ref cancel); DialogResult = DialogResult.Cancel; } bool wizardFinished = false; protected virtual void FinishWizard() { DialogResult = DialogResult.OK; Close(); Dispose(); } protected void FinishCanceled() { wizardFinished = false; WizardProgress_EnteringStep(null, new WizardProgressEventArgs(false)); } protected override void OnClosed(EventArgs e) { base.OnClosed(e); if (Owner != null) Owner.Focus(); } private void XenWizardBase_FormClosing(object sender, FormClosingEventArgs e) { if (wizardFinished) return; bool cancel = false; OnCancel(ref cancel); e.Cancel = cancel; } /// /// The help ID as sent to HelpManager when the user requests help on a given wizard page. /// The base implementation returns [typeName]_[CurrentStepTabPage.HelpID]Pane. /// Override this function or XenTabPage.HelpID if necessary (for example, some wizards have /// a single help ID for the whole wizard). /// protected virtual string WizardPaneHelpID() { return FormatHelpId(wizardProgress.CurrentStepTabPage.HelpID); } protected string FormatHelpId(string id) { return string.Format("{0}_{1}Pane", GetType().Name, id); } public bool HasHelp() { return HelpManager.TryGetTopicId(WizardPaneHelpID(), out _); } private void HelpButton_Click(object sender, EventArgs e) { Help.HelpManager.Launch(WizardPaneHelpID()); } private void XenWizardBase_HelpRequested(object sender, HelpEventArgs hlpevent) { Help.HelpManager.Launch(WizardPaneHelpID()); } private void XenWizardBase_Load(object sender, EventArgs e) { if (Owner == null) Owner = Program.MainWindow; CenterToParent(); FormFontFixer.Fix(this); if (wizardProgress.Steps.Count == 0) return; xenTabControlBody.SelectedTab = wizardProgress.Steps[0]; wizardProgress.CurrentStepTabPage.PageLoaded(PageLoadedDirection.Forward); UpdateWizard(); } private void XenWizardBase_KeyPress(object sender, KeyPressEventArgs e) { if (e.KeyChar == (char)Keys.Enter) { if (buttonNext.CanSelect) buttonNext.Select(); buttonNext.PerformClick(); } } [Browsable(false)] [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] public XenTabPage CurrentStepTabPage { get { return wizardProgress.CurrentStepTabPage; } } public void NextStep() { wizardProgress.NextStep(); } public void PreviousStep() { wizardProgress.PreviousStep(); } public bool IsFirstPage() { return wizardProgress.IsFirstStep; } protected void RefreshProgress() { wizardProgress.Refresh(); } private void SetTitle() { labelWizard.Text = wizardProgress.CurrentStepTabPage.PageTitle; } protected void UpdateWizard() { buttonPrevious.Enabled = !wizardProgress.IsFirstStep && wizardProgress.CurrentStepTabPage.EnablePrevious(); buttonNext.Enabled = wizardProgress.IsLastStep ? !wizardFinished && wizardProgress.CurrentStepTabPage.EnableNext() : wizardProgress.CurrentStepTabPage.EnableNext(); buttonCancel.Enabled = wizardProgress.CurrentStepTabPage.EnableCancel(); SetTitle(); buttonNext.Text = wizardProgress.CurrentStepTabPage.NextText(wizardProgress.IsLastStep); } private void page_StatusChanged(XenTabPage sender) { UpdateWizard(); } internal void DisablePage(XenTabPage wizardTabPage, bool disable) { foreach (XenTabPage p in wizardProgress.Steps) if (p == wizardTabPage) { p.DisableStep = disable; return; } } protected virtual IEnumerable> GetSummary() { var entries = new List>(); foreach (XenTabPage page in wizardProgress.Steps) entries.AddRange(page.PageSummary); return entries; } protected virtual void ShowInformationMessage(string message) { labelGeneralInformationMessage.Text = message; panelGeneralInformationMessage.Visible = true; } protected virtual void HideInformationMessage() { panelGeneralInformationMessage.Visible = false; } } }