Softex CelwareTech Blog
VSTO Officeアドイン2026-05-20

AvalonDockレイアウト保存が永続化されないバグをOnBeforeHideで防ぐ

VSTO の WinForms Form + ElementHost + WPF 構成で、Form.Hide() では WPF の Unloaded が発火せず AvalonDock レイアウトが保存されない問題への対策を解説します。

VSTOWPFAvalonDockWinFormsElementHost

はじめに

VSTO アドインで WinForms Form + ElementHost + WPF UserControl の構成を使い、内部に AvalonDock を置くことがあります。このとき、WPF 側の Unloaded イベントでレイアウト保存を書いていると、フォームを閉じたつもりなのに次回起動時にレイアウトが戻らないことがあります。

原因は、Excel と並行して使うために FormClosingClose() をキャンセルし、Form.Hide() に置き換えているケースです。Hide() では WPF の Unloaded が発火しないため、MainView.SaveLayout() が呼ばれません。

こんな場面で使えます

  • Excel VSTO アドインで独立フォーム型の WPF UI を出している
  • AvalonDock のペイン配置や ListView 列幅を保存したい
  • フォームの閉じるボタンやリボントグルでは Hide() している
  • ウィンドウ位置は保存されるのに、内部レイアウトだけ保存されない

解決策

フォーム側に OnBeforeHide コールバックを用意し、Hide() する直前に明示的に保存します。閉じるボタン、Alt+F4、リボントグル、Excel 終了の各経路から同じ保存メソッドを呼べる形にします。

using System;
using System.Windows.Forms;
using System.Windows.Forms.Integration;

internal sealed class MainToolForm : Form
{
    public MyMainView MainView { get; }
    public Action OnBeforeHide { get; set; }
    public bool AllowRealClose { get; set; }

    public MainToolForm(MyMainView mainView)
    {
        MainView = mainView ?? throw new ArgumentNullException(nameof(mainView));

        Controls.Add(new ElementHost
        {
            Dock = DockStyle.Fill,
            Child = mainView,
        });

        FormClosing += OnFormClosing;
    }

    private void OnFormClosing(object sender, FormClosingEventArgs e)
    {
        try
        {
            OnBeforeHide?.Invoke();
        }
        catch
        {
            // 保存失敗で終了処理を止めない
        }

        if (AllowRealClose)
        {
            return;
        }

        if (e.CloseReason == CloseReason.UserClosing)
        {
            e.Cancel = true;
            Hide();
        }
    }
}

VSTO 側では、フォーム位置と AvalonDock レイアウトをまとめて保存するメソッドを用意します。

private MainToolForm _mainForm;

private void OpenToolWindow()
{
    var mainView = new MyMainView();

    _mainForm = new MainToolForm(mainView)
    {
        OnBeforeHide = SaveAllState,
    };

    _mainForm.Show();
}

private void SaveAllState()
{
    SaveFormGeometry();

    try
    {
        _mainForm?.MainView?.SaveLayout();
    }
    catch
    {
        // レイアウト保存失敗で Excel を巻き込まない
    }
}

private void HideFromRibbon()
{
    if (_mainForm == null || !_mainForm.Visible) return;

    SaveAllState();
    _mainForm.Hide();
}

Excel 終了時に本当にフォームを閉じる場合も、同じ保存経路を通します。

private void ThisAddIn_Shutdown(object sender, EventArgs e)
{
    if (_mainForm == null) return;

    _mainForm.AllowRealClose = true;
    _mainForm.Close();
}

設計のポイント

保存処理は WPF 側の Unloaded だけに置かず、ホスト側の表示制御に合わせて呼び出します。VSTO では Excel を終了せずにツール画面だけを隠すことが多いため、Hide() が通常経路になります。

また、フォーム位置、サイズ、内部レイアウト、列幅などを SaveAllState() に集約すると、閉じるボタン、リボン、Excel 終了のどこからでも同じ保存処理を呼べます。

注意点・ハマりポイント

  • Form.Hide() では WPF の Unloaded が発火しない
  • ユーザー操作の Close()Hide() に変換している場合は、変換前に保存する
  • Excel 終了時は AllowRealClose = true で実際に閉じる
  • 保存失敗で Excel やアドインの終了を止めない
  • レイアウト保存とフォーム位置保存を別々の経路にしない
  • AvalonDock の新ペイン追加時は、保存済みレイアウト互換も確認する

関連記事

まとめ

VSTO の独立フォーム型 WPF UI では、Unloaded だけに保存を任せると Hide() 経路で漏れます。OnBeforeHide をフォーム側に用意し、非表示化の直前に SaveLayout() を明示的に呼ぶことで、AvalonDock のレイアウトを安定して永続化できます。

この技術で業務改善しませんか?

Excel VBA・GAS・Webアプリで業務の自動化ツールを開発しています。 「こんなことできる?」というご相談だけでもお気軽にどうぞ。

無料相談はこちら →