chromium/ash/system/session/logout_confirmation_controller.h

// Copyright 2014 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#ifndef ASH_SYSTEM_SESSION_LOGOUT_CONFIRMATION_CONTROLLER_H_
#define ASH_SYSTEM_SESSION_LOGOUT_CONFIRMATION_CONTROLLER_H_

#include <memory>

#include "ash/ash_export.h"
#include "ash/public/cpp/session/session_observer.h"
#include "base/functional/callback_forward.h"
#include "base/memory/raw_ptr.h"
#include "base/time/time.h"
#include "base/timer/timer.h"

class PrefRegistrySimple;

namespace base {
class TickClock;
}

namespace ash {

class LogoutConfirmationDialog;

// This class shows a dialog asking the user to confirm or deny logout and
// terminates the session if the user either confirms or allows the countdown
// shown in the dialog to expire.
//
// It is guaranteed that no more than one confirmation dialog will be visible at
// any given time. If there are multiple requests to show a confirmation dialog
// at the same time, the dialog whose countdown expires first is shown.
//
// In public sessions, asks the user to end the session when the last window is
// closed.
class ASH_EXPORT LogoutConfirmationController : public SessionObserver {
 public:
  enum class Source { kShelfExitButton, kCloseAllWindows };

  LogoutConfirmationController();

  LogoutConfirmationController(const LogoutConfirmationController&) = delete;
  LogoutConfirmationController& operator=(const LogoutConfirmationController&) =
      delete;

  ~LogoutConfirmationController() override;

  static void RegisterProfilePrefs(PrefRegistrySimple* registry);

  const base::TickClock* clock() const { return clock_; }

  // Shows a LogoutConfirmationDialog. If a confirmation dialog is already being
  // shown, it is closed and a new one opened if |logout_time| is earlier than
  // the current dialog's |logout_time_|.
  void ConfirmLogout(base::TimeTicks logout_time, Source source);

  // SessionObserver:
  void OnLoginStatusChanged(LoginStatus login_status) override;
  void OnLockStateChanged(bool locked) override;

  // Called by the |dialog_| when the user confirms logout.
  void OnLogoutConfirmed();

  // Called by the |dialog_| when it is closed.
  void OnDialogClosed();

  // Overrides the internal clock for testing. This doesn't take the ownership
  // of the clock. |clock| must outlive the LogoutConfirmationController
  // instance.
  void SetClockForTesting(const base::TickClock* clock);
  void SetLogoutCallbackForTesting(
      const base::RepeatingCallback<void(Source)>& logout_callback);
  LogoutConfirmationDialog* dialog_for_testing() const { return dialog_; }

  int confirm_logout_count_for_test() const {
    return confirm_logout_count_for_test_;
  }

 private:
  class LastWindowClosedObserver;
  std::unique_ptr<LastWindowClosedObserver> last_window_closed_observer_;

  raw_ptr<const base::TickClock> clock_;

  base::RepeatingCallback<void(Source)> logout_callback_;
  Source source_;

  base::TimeTicks logout_time_;
  raw_ptr<LogoutConfirmationDialog> dialog_ =
      nullptr;  // Owned by the Views hierarchy.
  base::OneShotTimer logout_timer_;

  int confirm_logout_count_for_test_ = 0;
};

}  // namespace ash

#endif  // ASH_SYSTEM_SESSION_LOGOUT_CONFIRMATION_CONTROLLER_H_