summaryrefslogtreecommitdiffstats
path: root/layout/printing/nsPrintData.cpp
blob: db09acae39513fe52990030189b6d4f903d62bc9 (plain)
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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 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 "nsPrintData.h"

#include "mozilla/gfx/PrintPromise.h"
#include "nsIStringBundle.h"
#include "nsIWidget.h"
#include "nsPrintObject.h"
#include "nsIWebProgressListener.h"
#include "mozilla/Services.h"

//-----------------------------------------------------
// PR LOGGING
#include "mozilla/Logging.h"

extern mozilla::LazyLogModule gPrintingLog;

#define PR_PL(_p1) MOZ_LOG(gPrintingLog, mozilla::LogLevel::Debug, _p1);

static void InformListenersOfProgressChange(
    const nsCOMArray<nsIWebProgressListener>& aListeners, int32_t aProgress,
    int32_t aMaxProgress, bool aDoStartStop, int32_t aFlag) {
  size_t numberOfListeners = aListeners.Length();
  for (size_t i = 0; i < numberOfListeners; ++i) {
    nsCOMPtr<nsIWebProgressListener> listener = aListeners.SafeElementAt(i);
    if (NS_WARN_IF(!listener)) {
      continue;
    }
    listener->OnProgressChange(nullptr, nullptr, aProgress, aMaxProgress,
                               aProgress, aMaxProgress);
    if (aDoStartStop) {
      listener->OnStateChange(nullptr, nullptr, aFlag, NS_OK);
    }
  }
}

static void InformListenersOfEndPrinting(
    const nsCOMArray<nsIWebProgressListener>& aListeners) {
  InformListenersOfProgressChange(
      aListeners, 100, 100, true,
      nsIWebProgressListener::STATE_STOP |
          nsIWebProgressListener::STATE_IS_DOCUMENT);
  InformListenersOfProgressChange(aListeners, 100, 100, true,
                                  nsIWebProgressListener::STATE_STOP |
                                      nsIWebProgressListener::STATE_IS_NETWORK);
}

//---------------------------------------------------
//-- nsPrintData Class Impl
//---------------------------------------------------
nsPrintData::nsPrintData(ePrintDataType aType)
    : mType(aType), mOnStartSent(false), mIsAborted(false) {}

nsPrintData::~nsPrintData() {
  // Two things need to be done:
  // - Inform the listeners
  // - End/Abort document
  // Preview requires neither, so return early.
  if (mType == eIsPrintPreview) {
    return;
  }

  if (mPrintDC) {
    PR_PL(("****************** End Document ************************\n"));
    PR_PL(("\n"));
    if (mPrintDC->IsCurrentlyPrintingDocument()) {
      if (!mIsAborted) {
        auto promise = mPrintDC->EndDocument();
        if (mOnStartSent) {
          promise->Then(mozilla::GetMainThreadSerialEventTarget(), __func__,
                        [listeners = std::move(mPrintProgressListeners)](
                            // We're in dtor, so capture listeners by move.
                            const mozilla::gfx::PrintEndDocumentPromise::
                                ResolveOrRejectValue&) {
                          InformListenersOfEndPrinting(listeners);
                        });
        }
        // Informing listeners asynchronously, or don't need to inform them, so
        // return early.
        return;
      }
      mPrintDC->AbortDocument();
    }
  }
  if (mOnStartSent) {
    // Synchronously notify the listeners.
    OnEndPrinting();
  }
}

void nsPrintData::OnStartPrinting() {
  if (!mOnStartSent) {
    InformListenersOfProgressChange(
        mPrintProgressListeners, 0, 0, true,
        nsIWebProgressListener::STATE_START |
            nsIWebProgressListener::STATE_IS_DOCUMENT |
            nsIWebProgressListener::STATE_IS_NETWORK);
    mOnStartSent = true;
  }
}

void nsPrintData::OnEndPrinting() {
  InformListenersOfEndPrinting(mPrintProgressListeners);
}

void nsPrintData::DoOnProgressChange(int32_t aProgress, int32_t aMaxProgress,
                                     bool aDoStartStop, int32_t aFlag) {
  InformListenersOfProgressChange(mPrintProgressListeners, aProgress,
                                  aMaxProgress, aDoStartStop, aFlag);
}

void nsPrintData::DoOnStatusChange(nsresult aStatus) {
  size_t numberOfListeners = mPrintProgressListeners.Length();
  for (size_t i = 0; i < numberOfListeners; ++i) {
    nsCOMPtr<nsIWebProgressListener> listener =
        mPrintProgressListeners.SafeElementAt(i);
    if (NS_WARN_IF(!listener)) {
      continue;
    }
    listener->OnStatusChange(nullptr, nullptr, aStatus, nullptr);
  }
}