File: PopupQueue.cpp

package info (click to toggle)
firefox 144.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 4,637,504 kB
  • sloc: cpp: 7,576,692; javascript: 6,430,831; ansic: 3,748,119; python: 1,398,978; xml: 628,810; asm: 438,679; java: 186,194; sh: 63,212; makefile: 19,159; objc: 13,086; perl: 12,986; yacc: 4,583; cs: 3,846; pascal: 3,448; lex: 1,720; ruby: 1,003; exp: 762; php: 436; lisp: 258; awk: 247; sql: 66; sed: 53; csh: 10
file content (111 lines) | stat: -rw-r--r-- 2,692 bytes parent folder | download | duplicates (5)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
/* -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 2; -*- */
/* vim: set sw=2 ts=8 et tw=80 : */
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

#include "PopupQueue.h"

#include "mozilla/StaticPtr.h"
#include "mozilla/dom/Element.h"
#include "nsThreadUtils.h"

using namespace mozilla;

void PopupQueue::Enqueue(Element* aPopup,
                         MoveOnlyFunction<void(Element*)>&& aCallback) {
  if (!aCallback) {
    return;
  }

  if (Store(aPopup, false, std::move(aCallback)) && !mShowing) {
    MaybeShowNext();
  }
}

void PopupQueue::Show(Element* aPopup,
                      MoveOnlyFunction<void(Element*)>&& aCallback) {
  if (!aCallback) {
    return;
  }

  MoveOnlyFunction<void(Element*)> callback = std::move(aCallback);

  if (Store(aPopup, true, nullptr)) {
    ++mShowing;
    callback(aPopup);
  }
}

bool PopupQueue::Store(Element* aPopup, bool aShown,
                       MoveOnlyFunction<void(Element*)>&& aCallback) {
  // Let's avoid the same popup shown multiple time, if it's already on screen.
  if (!aShown) {
    for (const PendingPopup& popup : mQueue) {
      if (popup.mPopup == aPopup) {
        return false;
      }
    }
  }

  mQueue.AppendElement(PendingPopup{aPopup,
                                    /* if not shown, it's qeueable */ !aShown,
                                    aShown, std::move(aCallback)});
  return true;
}

void PopupQueue::NotifyDismissed(Element* aPopup, bool aRemoveAll) {
  if (mQueue.IsEmpty()) {
    return;
  }

  for (uint32_t i = 0; i < mQueue.Length();) {
    if (mQueue[i].mPopup == aPopup) {
      if (mShowing && mQueue[i].mShown) {
        --mShowing;
      }

      mQueue.RemoveElementAt(i);

      if (!aRemoveAll) {
        break;
      }

      continue;
    }

    ++i;
  }

  if (!mQueue.IsEmpty() && !mShowing) {
    NS_DispatchToMainThread(NewRunnableMethod("PopupQueue::MaybeShowNext", this,
                                              &PopupQueue::MaybeShowNext));
  }
}

void PopupQueue::MaybeShowNext() {
  if (mQueue.IsEmpty() || mShowing) {
    return;
  }

  PendingPopup& popup = mQueue[0];

  ++mShowing;
  MOZ_ASSERT(!popup.mShown);
  popup.mShown = true;

  MoveOnlyFunction<void(Element*)> callback = std::move(popup.mCallback);
  MOZ_ASSERT(!popup.mCallback);

  callback(mQueue[0].mPopup);
}

PopupQueue::Element* PopupQueue::RetrieveQueueableShownPopup() const {
  for (auto& popup : mQueue) {
    if (popup.mQueueable && popup.mShown) {
      return popup.mPopup;
    }
  }

  return nullptr;
}