blob: 2e21e891ca99145d163fc9e2ba999d9a324238bc [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 "chrome/browser/enterprise/idle/dialog_manager.h"
#include <algorithm>
#include <utility>
#include "base/check.h"
#include "base/check_is_test.h"
#include "base/command_line.h"
#include "base/functional/bind.h"
#include "base/task/single_thread_task_runner.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_list.h"
#include "chrome/common/chrome_switches.h"
#include "components/enterprise/idle/idle_pref_names.h"
#include "components/enterprise/idle/metrics.h"
#include "components/prefs/pref_service.h"
namespace enterprise_idle {
namespace {
constexpr base::TimeDelta kTestDialogTimeout = base::Seconds(5);
constexpr base::TimeDelta kDialogTimeout = base::Seconds(30);
IdleDialog::ActionSet GetActionSet(PrefService* prefs) {
std::vector<ActionType> actions;
std::ranges::transform(prefs->GetList(prefs::kIdleTimeoutActions),
std::back_inserter(actions),
[](const base::Value& action) {
return static_cast<ActionType>(action.GetInt());
});
return ActionsToActionSet(base::flat_set<ActionType>(std::move(actions)));
}
} // namespace
DialogManager::DialogManager() = default;
DialogManager::~DialogManager() = default;
// static
DialogManager* DialogManager::GetInstance() {
static base::NoDestructor<DialogManager> instance;
return instance.get();
}
base::CallbackListSubscription DialogManager::MaybeShowDialog(
Profile* profile,
base::TimeDelta threshold,
const base::flat_set<ActionType>& action_types,
FinishedCallback on_finished) {
if (dialog_) {
// The dialog is already visible, re-use it.
return callbacks_.Add(std::move(on_finished));
}
Browser* active_browser = BrowserList::GetInstance()->GetLastActive();
if (!active_browser || !active_browser->IsActive() ||
!active_browser->is_type_normal()) {
// User is in another app, or in a window that shouldn't show the dialog
// (e.g. DevTools). Skip the dialog, and run actions immediately.
std::move(on_finished).Run(/*expired=*/true);
return base::CallbackListSubscription();
}
// Create a new dialog, modal to `active_browser`.
base::TimeDelta timeout = base::CommandLine::ForCurrentProcess()->HasSwitch(
switches::kSimulateIdleTimeout)
? kTestDialogTimeout
: kDialogTimeout;
dialog_timer_.Start(
FROM_HERE, timeout,
base::BindOnce(&DialogManager::OnDialogExpired, base::Unretained(this)));
dialog_ = IdleDialog::Show(
active_browser, timeout, threshold, GetActionSet(profile->GetPrefs()),
base::BindOnce(&DialogManager::OnDialogDismissedByUser,
base::Unretained(this)));
metrics::RecordIdleTimeoutDialogEvent(
metrics::IdleTimeoutDialogEvent::kDialogShown);
return callbacks_.Add(std::move(on_finished));
}
void DialogManager::DismissDialogForTesting() {
CHECK_IS_TEST();
OnDialogDismissedByUser();
}
bool DialogManager::IsDialogOpenForTesting() const {
return bool(dialog_);
}
void DialogManager::OnDialogDismissedByUser() {
if (dialog_) {
dialog_->Close();
}
dialog_.reset();
dialog_timer_.Stop();
metrics::RecordIdleTimeoutDialogEvent(
metrics::IdleTimeoutDialogEvent::kDialogDismissedByUser);
callbacks_.Notify(/*expired=*/false);
}
void DialogManager::OnDialogExpired() {
if (dialog_) {
dialog_->Close();
}
dialog_.reset();
dialog_timer_.Stop();
metrics::RecordIdleTimeoutDialogEvent(
metrics::IdleTimeoutDialogEvent::kDialogExpired);
callbacks_.Notify(/*expired=*/true);
}
} // namespace enterprise_idle