blob: 31f1e3de5c94389af1c8d19b148b7c6a107b4ce0 [file] [log] [blame]
// Copyright 2023 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "fuchsia_web/shell/present_frame.h"
#include <fuchsia/ui/policy/cpp/fidl.h>
#include <lib/sys/cpp/component_context.h>
#include <lib/sys/cpp/service_directory.h>
#include <lib/ui/scenic/cpp/view_ref_pair.h>
#include <lib/ui/scenic/cpp/view_token_pair.h>
#include <stdint.h>
#include <zircon/rights.h>
#include <memory>
#include <utility>
#include <vector>
#include "base/check.h"
#include "base/fuchsia/fuchsia_logging.h"
#include "base/fuchsia/process_context.h"
#include "base/logging.h"
fuchsia::ui::views::ViewRef CloneViewRef(
const fuchsia::ui::views::ViewRef& view_ref) {
fuchsia::ui::views::ViewRef dup;
zx_status_t status =
view_ref.reference.duplicate(ZX_RIGHT_SAME_RIGHTS, &dup.reference);
ZX_CHECK(status == ZX_OK, status) << "zx_object_duplicate";
return dup;
}
void PresentFrame(fuchsia::web::Frame* frame,
fidl::InterfaceHandle<fuchsia::element::AnnotationController>
annotation_controller) {
fuchsia::element::GraphicalPresenterSyncPtr presenter;
zx_status_t status = base::ComponentContextForProcess()->svc()->Connect(
presenter.NewRequest());
ZX_CHECK(status == ZX_OK, status)
<< "Couldn't connect to GraphicalPresenter.";
auto view_tokens = scenic::ViewTokenPair::New();
auto view_ref_pair = scenic::ViewRefPair::New();
fuchsia::element::ViewSpec view_spec;
view_spec.set_view_holder_token(std::move(view_tokens.view_holder_token));
view_spec.set_view_ref(CloneViewRef(view_ref_pair.view_ref));
view_spec.set_annotations({});
fuchsia::element::ViewControllerSyncPtr view_controller;
fuchsia::element::GraphicalPresenter_PresentView_Result present_view_result;
status = presenter->PresentView(
std::move(view_spec), std::move(annotation_controller),
view_controller.NewRequest(), &present_view_result);
// Note: We do not consider `present_view_result.is_err()` in the fallback
// condition in case the FIDL call succeeds but the method reports an error.
// This is because the only error type reported by the PresentView method is
// INVALID_ARGS, which we have carefully avoided by:
// * Providing a view_spec.view_holder_token and view_spec.view_ref (GFX)
// * Not providing _both_ GFX Views and Flatland Views at once.
//
// Therefore, we expect that if the FIDL call succeeds, the presentation
// should also succeed.
if (status == ZX_OK) {
DCHECK(!present_view_result.is_err())
<< "PresentView failed to display the view, reason: "
<< static_cast<uint32_t>(present_view_result.err());
} else {
// Fallback to connect to Root Presenter.
// TODO(http://crbug.com/1402457): Remove fallback.
LOG(INFO) << "PresentView failed to connect, reason: " << status
<< ". Falling back to fuchsia.ui.policy.Presenter.";
auto root_presenter = base::ComponentContextForProcess()
->svc()
->Connect<fuchsia::ui::policy::Presenter>();
// Replace the original ViewToken and ViewRefPair with new ones.
view_tokens = scenic::ViewTokenPair::New();
view_ref_pair = scenic::ViewRefPair::New();
root_presenter->PresentOrReplaceView2(
std::move(view_tokens.view_holder_token),
CloneViewRef(view_ref_pair.view_ref), nullptr);
}
// Present a fullscreen view of |frame|.
frame->CreateViewWithViewRef(std::move(view_tokens.view_token),
std::move(view_ref_pair.control_ref),
std::move(view_ref_pair.view_ref));
}