Coverage Report

Created: 2018-09-25 14:53

/src/mozilla-central/widget/nsFilePickerProxy.cpp
Line
Count
Source (jump to first uncovered line)
1
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
2
 *
3
 * This Source Code Form is subject to the terms of the Mozilla Public
4
 * License, v. 2.0. If a copy of the MPL was not distributed with this
5
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
6
7
#include "nsFilePickerProxy.h"
8
#include "nsComponentManagerUtils.h"
9
#include "nsIFile.h"
10
#include "nsSimpleEnumerator.h"
11
#include "mozilla/dom/Directory.h"
12
#include "mozilla/dom/File.h"
13
#include "mozilla/dom/TabChild.h"
14
#include "mozilla/dom/IPCBlobUtils.h"
15
16
using namespace mozilla::dom;
17
18
NS_IMPL_ISUPPORTS(nsFilePickerProxy, nsIFilePicker)
19
20
nsFilePickerProxy::nsFilePickerProxy()
21
  : mSelectedType(0)
22
  , mIPCActive(false)
23
0
{
24
0
}
25
26
nsFilePickerProxy::~nsFilePickerProxy()
27
0
{
28
0
}
29
30
NS_IMETHODIMP
31
nsFilePickerProxy::Init(mozIDOMWindowProxy* aParent, const nsAString& aTitle,
32
                        int16_t aMode)
33
0
{
34
0
  TabChild* tabChild = TabChild::GetFrom(aParent);
35
0
  if (!tabChild) {
36
0
    return NS_ERROR_FAILURE;
37
0
  }
38
0
39
0
  mParent = nsPIDOMWindowOuter::From(aParent);
40
0
41
0
  mMode = aMode;
42
0
43
0
  NS_ADDREF_THIS();
44
0
  tabChild->SendPFilePickerConstructor(this, nsString(aTitle), aMode);
45
0
46
0
  mIPCActive = true;
47
0
  return NS_OK;
48
0
}
49
50
void
51
nsFilePickerProxy::InitNative(nsIWidget* aParent, const nsAString& aTitle)
52
0
{
53
0
}
54
55
NS_IMETHODIMP
56
nsFilePickerProxy::AppendFilter(const nsAString& aTitle, const nsAString& aFilter)
57
0
{
58
0
  mFilterNames.AppendElement(aTitle);
59
0
  mFilters.AppendElement(aFilter);
60
0
  return NS_OK;
61
0
}
62
63
NS_IMETHODIMP
64
nsFilePickerProxy::GetDefaultString(nsAString& aDefaultString)
65
0
{
66
0
  aDefaultString = mDefault;
67
0
  return NS_OK;
68
0
}
69
70
NS_IMETHODIMP
71
nsFilePickerProxy::SetDefaultString(const nsAString& aDefaultString)
72
0
{
73
0
  mDefault = aDefaultString;
74
0
  return NS_OK;
75
0
}
76
77
NS_IMETHODIMP
78
nsFilePickerProxy::GetDefaultExtension(nsAString& aDefaultExtension)
79
0
{
80
0
  aDefaultExtension = mDefaultExtension;
81
0
  return NS_OK;
82
0
}
83
84
NS_IMETHODIMP
85
nsFilePickerProxy::SetDefaultExtension(const nsAString& aDefaultExtension)
86
0
{
87
0
  mDefaultExtension = aDefaultExtension;
88
0
  return NS_OK;
89
0
}
90
91
NS_IMETHODIMP
92
nsFilePickerProxy::GetFilterIndex(int32_t* aFilterIndex)
93
0
{
94
0
  *aFilterIndex = mSelectedType;
95
0
  return NS_OK;
96
0
}
97
98
NS_IMETHODIMP
99
nsFilePickerProxy::SetFilterIndex(int32_t aFilterIndex)
100
0
{
101
0
  mSelectedType = aFilterIndex;
102
0
  return NS_OK;
103
0
}
104
105
NS_IMETHODIMP
106
nsFilePickerProxy::GetFile(nsIFile** aFile)
107
0
{
108
0
  MOZ_ASSERT(false, "GetFile is unimplemented; use GetDomFileOrDirectory");
109
0
  return NS_ERROR_FAILURE;
110
0
}
111
112
NS_IMETHODIMP
113
nsFilePickerProxy::GetFileURL(nsIURI** aFileURL)
114
0
{
115
0
  MOZ_ASSERT(false, "GetFileURL is unimplemented; use GetDomFileOrDirectory");
116
0
  return NS_ERROR_FAILURE;
117
0
}
118
119
NS_IMETHODIMP
120
nsFilePickerProxy::GetFiles(nsISimpleEnumerator** aFiles)
121
0
{
122
0
  MOZ_ASSERT(false, "GetFiles is unimplemented; use GetDomFileOrDirectoryEnumerator");
123
0
  return NS_ERROR_FAILURE;
124
0
}
125
126
nsresult
127
nsFilePickerProxy::Show(int16_t* aReturn)
128
0
{
129
0
  MOZ_ASSERT(false, "Show is unimplemented; use Open");
130
0
  return NS_ERROR_NOT_IMPLEMENTED;
131
0
}
132
133
NS_IMETHODIMP
134
nsFilePickerProxy::Open(nsIFilePickerShownCallback* aCallback)
135
0
{
136
0
  mCallback = aCallback;
137
0
138
0
  nsString displayDirectory;
139
0
  if (mDisplayDirectory) {
140
0
    mDisplayDirectory->GetPath(displayDirectory);
141
0
  }
142
0
143
0
  if (!mIPCActive) {
144
0
    return NS_ERROR_FAILURE;
145
0
  }
146
0
147
0
  SendOpen(mSelectedType, mAddToRecentDocs, mDefault, mDefaultExtension,
148
0
           mFilters, mFilterNames, displayDirectory, mDisplaySpecialDirectory,
149
0
           mOkButtonLabel);
150
0
151
0
  return NS_OK;
152
0
}
153
154
mozilla::ipc::IPCResult
155
nsFilePickerProxy::Recv__delete__(const MaybeInputData& aData,
156
                                  const int16_t& aResult)
157
0
{
158
0
  if (aData.type() == MaybeInputData::TInputBlobs) {
159
0
    const InfallibleTArray<IPCBlob>& blobs = aData.get_InputBlobs().blobs();
160
0
    for (uint32_t i = 0; i < blobs.Length(); ++i) {
161
0
      RefPtr<BlobImpl> blobImpl = IPCBlobUtils::Deserialize(blobs[i]);
162
0
      NS_ENSURE_TRUE(blobImpl, IPC_OK());
163
0
164
0
      if (!blobImpl->IsFile()) {
165
0
        return IPC_OK();
166
0
      }
167
0
168
0
      nsPIDOMWindowInner* inner =
169
0
        mParent ? mParent->GetCurrentInnerWindow() : nullptr;
170
0
      RefPtr<File> file = File::Create(inner, blobImpl);
171
0
      MOZ_ASSERT(file);
172
0
173
0
      OwningFileOrDirectory* element = mFilesOrDirectories.AppendElement();
174
0
      element->SetAsFile() = file;
175
0
    }
176
0
  } else if (aData.type() == MaybeInputData::TInputDirectory) {
177
0
    nsCOMPtr<nsIFile> file;
178
0
    const nsAString& path(aData.get_InputDirectory().directoryPath());
179
0
    nsresult rv = NS_NewLocalFile(path, true, getter_AddRefs(file));
180
0
    if (NS_WARN_IF(NS_FAILED(rv))) {
181
0
      return IPC_OK();
182
0
    }
183
0
184
0
    RefPtr<Directory> directory =
185
0
      Directory::Create(mParent->GetCurrentInnerWindow(), file);
186
0
    MOZ_ASSERT(directory);
187
0
188
0
    OwningFileOrDirectory* element = mFilesOrDirectories.AppendElement();
189
0
    element->SetAsDirectory() = directory;
190
0
  }
191
0
192
0
  if (mCallback) {
193
0
    mCallback->Done(aResult);
194
0
    mCallback = nullptr;
195
0
  }
196
0
197
0
  return IPC_OK();
198
0
}
199
200
NS_IMETHODIMP
201
nsFilePickerProxy::GetDomFileOrDirectory(nsISupports** aValue)
202
0
{
203
0
  *aValue = nullptr;
204
0
  if (mFilesOrDirectories.IsEmpty()) {
205
0
    return NS_OK;
206
0
  }
207
0
208
0
  MOZ_ASSERT(mFilesOrDirectories.Length() == 1);
209
0
210
0
  if (mFilesOrDirectories[0].IsFile()) {
211
0
    nsCOMPtr<nsISupports> blob = ToSupports(mFilesOrDirectories[0].GetAsFile());
212
0
    blob.forget(aValue);
213
0
    return NS_OK;
214
0
  }
215
0
216
0
  MOZ_ASSERT(mFilesOrDirectories[0].IsDirectory());
217
0
  RefPtr<Directory> directory = mFilesOrDirectories[0].GetAsDirectory();
218
0
  directory.forget(aValue);
219
0
  return NS_OK;
220
0
}
221
222
namespace {
223
224
class SimpleEnumerator final : public nsSimpleEnumerator
225
{
226
public:
227
  explicit
228
  SimpleEnumerator(const nsTArray<OwningFileOrDirectory>& aFilesOrDirectories)
229
    : mFilesOrDirectories(aFilesOrDirectories)
230
    , mIndex(0)
231
0
  {}
232
233
  NS_IMETHOD
234
  HasMoreElements(bool* aRetvalue) override
235
0
  {
236
0
    MOZ_ASSERT(aRetvalue);
237
0
    *aRetvalue = mIndex < mFilesOrDirectories.Length();
238
0
    return NS_OK;
239
0
  }
240
241
  NS_IMETHOD
242
  GetNext(nsISupports** aValue) override
243
0
  {
244
0
    NS_ENSURE_TRUE(mIndex < mFilesOrDirectories.Length(), NS_ERROR_FAILURE);
245
0
246
0
    uint32_t index = mIndex++;
247
0
248
0
    if (mFilesOrDirectories[index].IsFile()) {
249
0
      nsCOMPtr<nsISupports> blob = ToSupports(mFilesOrDirectories[index].GetAsFile());
250
0
      blob.forget(aValue);
251
0
      return NS_OK;
252
0
    }
253
0
254
0
    MOZ_ASSERT(mFilesOrDirectories[index].IsDirectory());
255
0
    RefPtr<Directory> directory = mFilesOrDirectories[index].GetAsDirectory();
256
0
    directory.forget(aValue);
257
0
    return NS_OK;
258
0
  }
259
260
private:
261
  nsTArray<mozilla::dom::OwningFileOrDirectory> mFilesOrDirectories;
262
  uint32_t mIndex;
263
};
264
265
} // namespace
266
267
NS_IMETHODIMP
268
nsFilePickerProxy::GetDomFileOrDirectoryEnumerator(nsISimpleEnumerator** aDomfiles)
269
0
{
270
0
  RefPtr<SimpleEnumerator> enumerator =
271
0
    new SimpleEnumerator(mFilesOrDirectories);
272
0
  enumerator.forget(aDomfiles);
273
0
  return NS_OK;
274
0
}
275
276
void
277
nsFilePickerProxy::ActorDestroy(ActorDestroyReason aWhy)
278
0
{
279
0
  mIPCActive = false;
280
0
281
0
  if (mCallback) {
282
0
    mCallback->Done(nsIFilePicker::returnCancel);
283
0
    mCallback = nullptr;
284
0
  }
285
0
}