// Copyright © 2021 The CefSharp Authors. All rights reserved.
//
// Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
using System.Collections.Generic;
namespace CefSharp.Handler
{
///
/// Inherit from this class to handle frame events
/// All methods will be called on the CEF UI thread
///
public class FrameHandler : IFrameHandler
{
private Dictionary frames = new Dictionary();
///
void IFrameHandler.OnFrameAttached(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame, bool reattached)
{
//If we have a reference to the frame then we'll reuse that for memory management purposes
//The frame param massed on here is stack allocated so will be disposed when out of scope.
var storedFrame = GetFrameById(frame.Identifier);
OnFrameAttached(chromiumWebBrowser, browser, storedFrame ?? frame, reattached);
}
///
/// Called when a frame can begin routing commands to/from the associated
/// renderer process. Any commands that were queued have now been dispatched.
///
/// the ChromiumWebBrowser control
/// the browser object
/// the frame object
/// will be true if the frame was re-attached after exiting the BackForwardCache.
protected virtual void OnFrameAttached(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame, bool reattached)
{
}
///
void IFrameHandler.OnFrameCreated(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame)
{
frames.Add(frame.Identifier, frame);
OnFrameCreated(chromiumWebBrowser, browser, frame);
}
///
/// Called when a new frame is created. This will be the first notification
/// that references . Any commands that require transport to the
/// associated renderer process (LoadRequest, SendProcessMessage, GetSource,
/// etc.) will be queued until OnFrameAttached is called for .
///
/// the ChromiumWebBrowser control
/// the browser object
/// the frame object
protected virtual void OnFrameCreated(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame)
{
}
///
void IFrameHandler.OnFrameDetached(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame)
{
frames.Remove(frame.Identifier);
OnFrameDetached(chromiumWebBrowser, browser, frame);
}
///
/// Called when a frame loses its connection to the renderer process and will
/// be destroyed. Any pending or future commands will be discarded and
/// will now return false for . If called after
/// during browser destruction then
/// will return false for .
///
/// the ChromiumWebBrowser control
/// the browser object
/// the frame object
protected virtual void OnFrameDetached(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame frame)
{
}
///
void IFrameHandler.OnMainFrameChanged(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame oldFrame, IFrame newFrame)
{
//If we have a reference to the frame then we'll reuse that for memory management purposes
//The frame param massed on here is stack allocated so will be disposed when out of scope.
var storedFrame = newFrame == null ? null : GetFrameById(newFrame.Identifier);
OnMainFrameChanged(chromiumWebBrowser, browser, oldFrame, storedFrame ?? newFrame);
}
///
/// Called when the main frame changes due to one of the following:
/// - (a) initial browser creation
/// - (b) final browser destruction
/// - (c) cross-origin navigation
/// - (d) re-navigation after renderer process termination (due to crashes, etc).
///
/// will be null and will be non-null when a main frame is assigned
/// to for the first time.
/// will be non-null and will be null when a main frame is
/// removed from for the last time.
/// Both and will be non-nullfor cross-origin
/// navigations or re-navigation after renderer process termination.
/// This method will be called after for and/or after
/// for . If called after
/// during browser destruction then
/// will return false for .
///
/// the ChromiumWebBrowser control
/// the browser object
/// the old frame object
/// the new frame object
protected virtual void OnMainFrameChanged(IWebBrowser chromiumWebBrowser, IBrowser browser, IFrame oldFrame, IFrame newFrame)
{
}
private IFrame GetFrameById(string frameId)
{
if(frames.TryGetValue(frameId, out var frame))
{
return frame;
}
return null;
}
}
}