// 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; } } }