1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #import <Cocoa/Cocoa.h>
7 #import "base/mac/scoped_block.h"
8 #import "base/mac/scoped_nsobject.h"
9 #include "base/memory/scoped_ptr.h"
10 #include "chrome/browser/download/download_shelf.h"
11 #include "chrome/browser/ui/cocoa/cocoa_profile_test.h"
12 #import "chrome/browser/ui/cocoa/download/download_item_controller.h"
13 #import "chrome/browser/ui/cocoa/download/download_shelf_controller.h"
14 #import "chrome/browser/ui/cocoa/view_resizer_pong.h"
15 #include "content/public/test/mock_download_item.h"
16 #include "testing/gtest/include/gtest/gtest.h"
17 #include "testing/platform_test.h"
18 #import "third_party/ocmock/OCMock/OCMock.h"
19 #import "third_party/ocmock/gtest_support.h"
21 using ::testing::Return;
22 using ::testing::AnyNumber;
24 // Wraps a content::MockDownloadItem so it can be retained by the mock
25 // DownloadItemController.
26 @interface WrappedMockDownloadItem : NSObject {
28 scoped_ptr<content::MockDownloadItem> download_;
30 - (id)initWithMockDownload:(scoped_ptr<content::MockDownloadItem>)download;
31 - (content::DownloadItem*)download;
32 - (content::MockDownloadItem*)mockDownload;
35 @implementation WrappedMockDownloadItem
36 - (id)initWithMockDownload:(scoped_ptr<content::MockDownloadItem>)download {
37 if ((self = [super init])) {
38 download_ = download.Pass();
43 - (content::DownloadItem*)download {
44 return download_.get();
47 - (content::MockDownloadItem*)mockDownload {
48 return download_.get();
52 // Test method for accessing the wrapped MockDownloadItem.
53 @interface DownloadItemController (DownloadShelfControllerTest) {
55 - (WrappedMockDownloadItem*)wrappedMockDownload;
58 @implementation DownloadItemController (DownloadShelfControllerTest)
59 - (WrappedMockDownloadItem*)wrappedMockDownload {
64 @interface DownloadShelfController (Testing)
65 - (void)animationDidEnd:(NSAnimation*)animation;
68 // Subclass of the DownloadShelfController to override scheduleAutoClose and
69 // cancelAutoClose. During regular operation, a scheduled autoClose waits for 5
70 // seconds before closing the shelf (unless it is cancelled during this
71 // time). For testing purposes, we count the number of invocations of
72 // {schedule,cancel}AutoClose instead of actually scheduling and cancelling.
73 @interface CountingDownloadShelfController : DownloadShelfController {
75 int scheduleAutoCloseCount_;
76 int cancelAutoCloseCount_;
77 base::mac::ScopedBlock<dispatch_block_t> closeAnimationHandler_;
80 // Handler will be called at the end of a close animation.
81 - (void)setCloseAnimationHandler:(dispatch_block_t)handler;
84 @implementation CountingDownloadShelfController
86 -(void)scheduleAutoClose {
87 ++scheduleAutoCloseCount_;
90 -(void)cancelAutoClose {
91 ++cancelAutoCloseCount_;
94 - (void)setCloseAnimationHandler:(dispatch_block_t)handler {
95 closeAnimationHandler_.reset(handler);
98 - (void)animationDidEnd:(NSAnimation*)animation {
99 [super animationDidEnd:animation];
100 if (closeAnimationHandler_)
101 closeAnimationHandler_.get()();
108 class DownloadShelfControllerTest : public CocoaProfileTest {
110 void SetUp() override {
111 CocoaProfileTest::SetUp();
112 ASSERT_TRUE(browser());
114 resize_delegate_.reset([[ViewResizerPong alloc] init]);
115 shelf_.reset([[CountingDownloadShelfController alloc]
116 initWithBrowser:browser()
117 resizeDelegate:resize_delegate_.get()]);
118 EXPECT_TRUE([shelf_ view]);
119 [[test_window() contentView] addSubview:[shelf_ view]];
122 void TearDown() override {
126 CocoaProfileTest::TearDown();
130 id CreateItemController();
132 base::scoped_nsobject<CountingDownloadShelfController> shelf_;
133 base::scoped_nsobject<ViewResizerPong> resize_delegate_;
136 id DownloadShelfControllerTest::CreateItemController() {
137 scoped_ptr<content::MockDownloadItem> download(
138 new ::testing::NiceMock<content::MockDownloadItem>);
139 ON_CALL(*download.get(), GetOpened())
140 .WillByDefault(Return(false));
141 ON_CALL(*download.get(), GetState())
142 .WillByDefault(Return(content::DownloadItem::IN_PROGRESS));
144 base::scoped_nsobject<WrappedMockDownloadItem> wrappedMockDownload(
145 [[WrappedMockDownloadItem alloc] initWithMockDownload:download.Pass()]);
148 [OCMockObject mockForClass:[DownloadItemController class]];
149 base::scoped_nsobject<NSView> view([[NSView alloc] initWithFrame:NSZeroRect]);
150 [[[item_controller stub] andCall:@selector(download)
151 onObject:wrappedMockDownload.get()] download];
152 [[item_controller stub] updateVisibility:[OCMArg any]];
153 [[[item_controller stub]
154 andReturnValue:[NSValue valueWithSize:NSMakeSize(10,10)]] preferredSize];
155 [[[item_controller stub] andReturn:view.get()] view];
156 [[[item_controller stub]
157 andReturn:wrappedMockDownload.get()] wrappedMockDownload];
158 return [item_controller retain];
161 TEST_VIEW(DownloadShelfControllerTest, [shelf_ view]);
163 // Removing the last download from the shelf should cause it to close
165 TEST_F(DownloadShelfControllerTest, AddAndRemoveDownload) {
166 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
167 [shelf_ showDownloadShelf:YES
169 EXPECT_TRUE([shelf_ isVisible]);
170 EXPECT_TRUE([shelf_ bridge]->IsShowing());
172 [shelf_ remove:item];
173 EXPECT_FALSE([shelf_ isVisible]);
174 EXPECT_FALSE([shelf_ bridge]->IsShowing());
175 // The shelf should be closed without scheduling an autoClose.
176 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
179 // Test that the shelf doesn't close automatically after a removal if there are
180 // active download items still on the shelf.
181 TEST_F(DownloadShelfControllerTest, AddAndRemoveWithActiveItem) {
182 base::scoped_nsobject<DownloadItemController> item1(CreateItemController());
183 base::scoped_nsobject<DownloadItemController> item2(CreateItemController());
184 [shelf_ showDownloadShelf:YES
186 EXPECT_TRUE([shelf_ isVisible]);
187 [shelf_ add:item1.get()];
188 [shelf_ add:item2.get()];
189 [shelf_ remove:item1.get()];
190 EXPECT_TRUE([shelf_ isVisible]);
191 [shelf_ remove:item2.get()];
192 EXPECT_FALSE([shelf_ isVisible]);
193 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
196 // DownloadShelf::Unhide() should cause the shelf to be displayed if there are
197 // active downloads on it.
198 TEST_F(DownloadShelfControllerTest, HideAndUnhide) {
199 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
200 [shelf_ showDownloadShelf:YES
202 EXPECT_TRUE([shelf_ isVisible]);
203 [shelf_ add:item.get()];
204 [shelf_ bridge]->Hide();
205 EXPECT_FALSE([shelf_ isVisible]);
206 [shelf_ bridge]->Unhide();
207 EXPECT_TRUE([shelf_ isVisible]);
208 [shelf_ remove:item.get()];
209 EXPECT_FALSE([shelf_ isVisible]);
212 // DownloadShelf::Unhide() shouldn't cause the shelf to be displayed if all
213 // active downloads are removed from the shelf while the shelf was hidden.
214 TEST_F(DownloadShelfControllerTest, HideAutocloseUnhide) {
215 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
216 [shelf_ showDownloadShelf:YES
218 EXPECT_TRUE([shelf_ isVisible]);
219 [shelf_ add:item.get()];
220 [shelf_ bridge]->Hide();
221 EXPECT_FALSE([shelf_ isVisible]);
222 [shelf_ remove:item.get()];
223 EXPECT_FALSE([shelf_ isVisible]);
224 [shelf_ bridge]->Unhide();
225 EXPECT_FALSE([shelf_ isVisible]);
228 // Test of autoclosing behavior after opening a download item. The mouse is on
229 // the download shelf at the time the autoclose is scheduled.
230 TEST_F(DownloadShelfControllerTest, AutoCloseAfterOpenWithMouseInShelf) {
231 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
232 [shelf_ showDownloadShelf:YES
234 EXPECT_TRUE([shelf_ isVisible]);
235 [shelf_ add:item.get()];
236 // Expect 2 cancelAutoClose calls: From the showDownloadShelf: call and the
238 EXPECT_EQ(2, shelf_.get()->cancelAutoCloseCount_);
239 shelf_.get()->cancelAutoCloseCount_ = 0;
241 // The mouse enters the shelf.
242 [shelf_ mouseEntered:nil];
243 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
245 // The download opens.
246 EXPECT_CALL(*[[item wrappedMockDownload] mockDownload], GetOpened())
247 .WillRepeatedly(Return(true));
248 [shelf_ downloadWasOpened:item.get()];
250 // The shelf should now be waiting for the mouse to exit.
251 EXPECT_TRUE([shelf_ isVisible]);
252 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
253 EXPECT_EQ(0, shelf_.get()->cancelAutoCloseCount_);
255 // The mouse exits the shelf. autoClose should be scheduled now.
256 [shelf_ mouseExited:nil];
257 EXPECT_EQ(1, shelf_.get()->scheduleAutoCloseCount_);
258 EXPECT_EQ(0, shelf_.get()->cancelAutoCloseCount_);
260 // The mouse enters the shelf again. The autoClose should be cancelled.
261 [shelf_ mouseEntered:nil];
262 EXPECT_EQ(1, shelf_.get()->scheduleAutoCloseCount_);
263 EXPECT_EQ(1, shelf_.get()->cancelAutoCloseCount_);
266 // Test of autoclosing behavior after opening a download item.
267 TEST_F(DownloadShelfControllerTest, AutoCloseAfterOpenWithMouseOffShelf) {
268 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
269 [shelf_ showDownloadShelf:YES
271 EXPECT_TRUE([shelf_ isVisible]);
272 [shelf_ add:item.get()];
274 // The download is opened.
275 EXPECT_CALL(*[[item wrappedMockDownload] mockDownload], GetOpened())
276 .WillRepeatedly(Return(true));
277 [shelf_ downloadWasOpened:item.get()];
279 // The shelf should be closed immediately since the mouse is not over the
281 EXPECT_FALSE([shelf_ isVisible]);
284 // Test that if the shelf is closed while an autoClose is pending, the pending
285 // autoClose is cancelled.
286 TEST_F(DownloadShelfControllerTest, CloseWithPendingAutoClose) {
287 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
288 [shelf_ showDownloadShelf:YES
290 EXPECT_TRUE([shelf_ isVisible]);
291 [shelf_ add:item.get()];
292 // Expect 2 cancelAutoClose calls: From the showDownloadShelf: call and the
294 EXPECT_EQ(2, shelf_.get()->cancelAutoCloseCount_);
295 shelf_.get()->cancelAutoCloseCount_ = 0;
297 // The mouse enters the shelf.
298 [shelf_ mouseEntered:nil];
299 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
301 // The download opens.
302 EXPECT_CALL(*[[item wrappedMockDownload] mockDownload], GetOpened())
303 .WillRepeatedly(Return(true));
304 [shelf_ downloadWasOpened:item.get()];
306 // The shelf should now be waiting for the mouse to exit. autoClose should not
308 EXPECT_TRUE([shelf_ isVisible]);
309 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
310 EXPECT_EQ(0, shelf_.get()->cancelAutoCloseCount_);
312 // The mouse exits the shelf. autoClose should be scheduled now.
313 [shelf_ mouseExited:nil];
314 EXPECT_EQ(1, shelf_.get()->scheduleAutoCloseCount_);
315 EXPECT_EQ(0, shelf_.get()->cancelAutoCloseCount_);
317 // Remove the download item. This should cause the download shelf to be hidden
318 // immediately. The pending autoClose should be cancelled.
319 [shelf_ remove:item];
320 EXPECT_EQ(1, shelf_.get()->scheduleAutoCloseCount_);
321 EXPECT_EQ(1, shelf_.get()->cancelAutoCloseCount_);
322 EXPECT_FALSE([shelf_ isVisible]);
325 // That that the shelf cancels a pending autoClose if a new download item is
327 TEST_F(DownloadShelfControllerTest, AddItemWithPendingAutoClose) {
328 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
329 [shelf_ showDownloadShelf:YES
331 EXPECT_TRUE([shelf_ isVisible]);
332 [shelf_ add:item.get()];
333 // Expect 2 cancelAutoClose calls: From the showDownloadShelf: call and the
335 EXPECT_EQ(2, shelf_.get()->cancelAutoCloseCount_);
336 shelf_.get()->cancelAutoCloseCount_ = 0;
338 // The mouse enters the shelf.
339 [shelf_ mouseEntered:nil];
340 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
342 // The download opens.
343 EXPECT_CALL(*[[item wrappedMockDownload] mockDownload], GetOpened())
344 .WillRepeatedly(Return(true));
345 [shelf_ downloadWasOpened:item.get()];
347 // The shelf should now be waiting for the mouse to exit. autoClose should not
349 EXPECT_TRUE([shelf_ isVisible]);
350 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
351 EXPECT_EQ(0, shelf_.get()->cancelAutoCloseCount_);
353 // The mouse exits the shelf. autoClose should be scheduled now.
354 [shelf_ mouseExited:nil];
355 EXPECT_EQ(1, shelf_.get()->scheduleAutoCloseCount_);
356 EXPECT_EQ(0, shelf_.get()->cancelAutoCloseCount_);
358 // Add a new download item. The pending autoClose should be cancelled.
359 base::scoped_nsobject<DownloadItemController> item2(CreateItemController());
360 [shelf_ add:item.get()];
361 EXPECT_EQ(1, shelf_.get()->scheduleAutoCloseCount_);
362 EXPECT_EQ(1, shelf_.get()->cancelAutoCloseCount_);
363 EXPECT_TRUE([shelf_ isVisible]);
366 // Test that pending autoClose calls are cancelled when exiting.
367 TEST_F(DownloadShelfControllerTest, CancelAutoCloseOnExit) {
368 base::scoped_nsobject<DownloadItemController> item(CreateItemController());
369 [shelf_ showDownloadShelf:YES
371 EXPECT_TRUE([shelf_ isVisible]);
372 [shelf_ add:item.get()];
373 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
374 EXPECT_EQ(2, shelf_.get()->cancelAutoCloseCount_);
376 [shelf_ browserWillBeDestroyed];
377 EXPECT_EQ(0, shelf_.get()->scheduleAutoCloseCount_);
378 EXPECT_EQ(3, shelf_.get()->cancelAutoCloseCount_);
382 // The view should not be hidden when the shelf is shown.
383 // The view should be hidden after the closing animation.
384 TEST_F(DownloadShelfControllerTest, ViewVisibility) {
385 [shelf_ showDownloadShelf:YES isUserAction:NO];
386 EXPECT_FALSE([[shelf_ view] isHidden]);
388 [shelf_ setCloseAnimationHandler:^{
389 base::MessageLoop::current()->QuitNow();
391 [shelf_ showDownloadShelf:NO isUserAction:NO];
392 base::MessageLoop::current()->Run();
393 EXPECT_TRUE([[shelf_ view] isHidden]);
395 [shelf_ showDownloadShelf:YES isUserAction:NO];
396 EXPECT_FALSE([[shelf_ view] isHidden]);