1// Copyright (c) 2012 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.
4
5#include "ui/app_list/views/apps_grid_view.h"
6
7#include <algorithm>
8#include <set>
9#include <string>
10
11#include "base/guid.h"
12#include "ui/app_list/app_list_constants.h"
13#include "ui/app_list/app_list_folder_item.h"
14#include "ui/app_list/app_list_item.h"
15#include "ui/app_list/app_list_switches.h"
16#include "ui/app_list/pagination_controller.h"
17#include "ui/app_list/views/app_list_drag_and_drop_host.h"
18#include "ui/app_list/views/app_list_folder_view.h"
19#include "ui/app_list/views/app_list_item_view.h"
20#include "ui/app_list/views/apps_grid_view_delegate.h"
21#include "ui/app_list/views/page_switcher.h"
22#include "ui/app_list/views/pulsing_block_view.h"
23#include "ui/app_list/views/top_icon_animation_view.h"
24#include "ui/compositor/scoped_layer_animation_settings.h"
25#include "ui/events/event.h"
26#include "ui/gfx/animation/animation.h"
27#include "ui/gfx/geometry/vector2d.h"
28#include "ui/gfx/geometry/vector2d_conversions.h"
29#include "ui/views/border.h"
30#include "ui/views/view_model_utils.h"
31#include "ui/views/widget/widget.h"
32
33#if defined(USE_AURA)
34#include "ui/aura/window.h"
35#include "ui/aura/window_event_dispatcher.h"
36#if defined(OS_WIN)
37#include "ui/views/win/hwnd_util.h"
38#endif  // defined(OS_WIN)
39#endif  // defined(USE_AURA)
40
41#if defined(OS_WIN)
42#include "base/command_line.h"
43#include "base/files/file_path.h"
44#include "base/win/shortcut.h"
45#include "ui/base/dragdrop/drag_utils.h"
46#include "ui/base/dragdrop/drop_target_win.h"
47#include "ui/base/dragdrop/os_exchange_data.h"
48#include "ui/base/dragdrop/os_exchange_data_provider_win.h"
49#include "ui/gfx/win/dpi.h"
50#endif
51
52namespace app_list {
53
54namespace {
55
56// Distance a drag needs to be from the app grid to be considered 'outside', at
57// which point we rearrange the apps to their pre-drag configuration, as a drop
58// then would be canceled. We have a buffer to make it easier to drag apps to
59// other pages.
60const int kDragBufferPx = 20;
61
62// Padding space in pixels for fixed layout.
63const int kLeftRightPadding = 23;
64const int kTopPadding = 1;
65
66// Padding space in pixels between pages.
67const int kPagePadding = 40;
68
69// Preferred tile size when showing in fixed layout.
70const int kPreferredTileWidth = 88;
71const int kPreferredTileHeight = 98;
72
73const int kExperimentalPreferredTileWidth = 90;
74const int kExperimentalPrefferedTileHeight = 90;
75
76// Padding on each side of a tile.
77const int kExperimentalTileLeftRightPadding = 15;
78const int kExperimentalTileTopBottomPadding = 11;
79
80// Width in pixels of the area on the sides that triggers a page flip.
81const int kPageFlipZoneSize = 40;
82
83// Delay in milliseconds to do the page flip.
84const int kPageFlipDelayInMs = 1000;
85
86// How many pages on either side of the selected one we prerender.
87const int kPrerenderPages = 1;
88
89// The drag and drop proxy should get scaled by this factor.
90const float kDragAndDropProxyScale = 1.5f;
91
92// Delays in milliseconds to show folder dropping preview circle.
93const int kFolderDroppingDelay = 150;
94
95// Delays in milliseconds to show re-order preview.
96const int kReorderDelay = 120;
97
98// Delays in milliseconds to show folder item reparent UI.
99const int kFolderItemReparentDelay = 50;
100
101// Radius of the circle, in which if entered, show folder dropping preview
102// UI.
103const int kFolderDroppingCircleRadius = 39;
104
105// Returns the size of a tile view excluding its padding.
106gfx::Size GetTileViewSize() {
107  return switches::IsExperimentalAppListEnabled()
108             ? gfx::Size(kExperimentalPreferredTileWidth,
109                         kExperimentalPrefferedTileHeight)
110             : gfx::Size(kPreferredTileWidth, kPreferredTileHeight);
111}
112
113// Returns the size of a tile view inccluding its padding.
114gfx::Size GetTotalTileSize() {
115  gfx::Size size = GetTileViewSize();
116  if (switches::IsExperimentalAppListEnabled())
117    size.Enlarge(2 * kExperimentalTileLeftRightPadding,
118                 2 * kExperimentalTileTopBottomPadding);
119  return size;
120}
121
122// RowMoveAnimationDelegate is used when moving an item into a different row.
123// Before running the animation, the item's layer is re-created and kept in
124// the original position, then the item is moved to just before its target
125// position and opacity set to 0. When the animation runs, this delegate moves
126// the layer and fades it out while fading in the item at the same time.
127class RowMoveAnimationDelegate : public gfx::AnimationDelegate {
128 public:
129  RowMoveAnimationDelegate(views::View* view,
130                           ui::Layer* layer,
131                           const gfx::Rect& layer_target)
132      : view_(view),
133        layer_(layer),
134        layer_start_(layer ? layer->bounds() : gfx::Rect()),
135        layer_target_(layer_target) {
136  }
137  virtual ~RowMoveAnimationDelegate() {}
138
139  // gfx::AnimationDelegate overrides:
140  virtual void AnimationProgressed(const gfx::Animation* animation) OVERRIDE {
141    view_->layer()->SetOpacity(animation->GetCurrentValue());
142    view_->layer()->ScheduleDraw();
143
144    if (layer_) {
145      layer_->SetOpacity(1 - animation->GetCurrentValue());
146      layer_->SetBounds(animation->CurrentValueBetween(layer_start_,
147                                                       layer_target_));
148      layer_->ScheduleDraw();
149    }
150  }
151  virtual void AnimationEnded(const gfx::Animation* animation) OVERRIDE {
152    view_->layer()->SetOpacity(1.0f);
153    view_->SchedulePaint();
154  }
155  virtual void AnimationCanceled(const gfx::Animation* animation) OVERRIDE {
156    view_->layer()->SetOpacity(1.0f);
157    view_->SchedulePaint();
158  }
159
160 private:
161  // The view that needs to be wrapped. Owned by views hierarchy.
162  views::View* view_;
163
164  scoped_ptr<ui::Layer> layer_;
165  const gfx::Rect layer_start_;
166  const gfx::Rect layer_target_;
167
168  DISALLOW_COPY_AND_ASSIGN(RowMoveAnimationDelegate);
169};
170
171// ItemRemoveAnimationDelegate is used to show animation for removing an item.
172// This happens when user drags an item into a folder. The dragged item will
173// be removed from the original list after it is dropped into the folder.
174class ItemRemoveAnimationDelegate : public gfx::AnimationDelegate {
175 public:
176  explicit ItemRemoveAnimationDelegate(views::View* view)
177      : view_(view) {
178  }
179
180  virtual ~ItemRemoveAnimationDelegate() {
181  }
182
183  // gfx::AnimationDelegate overrides:
184  virtual void AnimationProgressed(const gfx::Animation* animation) OVERRIDE {
185    view_->layer()->SetOpacity(1 - animation->GetCurrentValue());
186    view_->layer()->ScheduleDraw();
187  }
188
189 private:
190  scoped_ptr<views::View> view_;
191
192  DISALLOW_COPY_AND_ASSIGN(ItemRemoveAnimationDelegate);
193};
194
195// ItemMoveAnimationDelegate observes when an item finishes animating when it is
196// not moving between rows. This is to ensure an item is repainted for the
197// "zoom out" case when releasing an item being dragged.
198class ItemMoveAnimationDelegate : public gfx::AnimationDelegate {
199 public:
200  ItemMoveAnimationDelegate(views::View* view) : view_(view) {}
201
202  virtual void AnimationEnded(const gfx::Animation* animation) OVERRIDE {
203    view_->SchedulePaint();
204  }
205  virtual void AnimationCanceled(const gfx::Animation* animation) OVERRIDE {
206    view_->SchedulePaint();
207  }
208
209 private:
210  views::View* view_;
211
212  DISALLOW_COPY_AND_ASSIGN(ItemMoveAnimationDelegate);
213};
214
215// Returns true if the |item| is an folder item.
216bool IsFolderItem(AppListItem* item) {
217  return (item->GetItemType() == AppListFolderItem::kItemType);
218}
219
220bool IsOEMFolderItem(AppListItem* item) {
221  return IsFolderItem(item) &&
222         (static_cast<AppListFolderItem*>(item))->folder_type() ==
223             AppListFolderItem::FOLDER_TYPE_OEM;
224}
225
226int ClampToRange(int value, int min, int max) {
227  return std::min(std::max(value, min), max);
228}
229
230}  // namespace
231
232#if defined(OS_WIN)
233// Interprets drag events sent from Windows via the drag/drop API and forwards
234// them to AppsGridView.
235// On Windows, in order to have the OS perform the drag properly we need to
236// provide it with a shortcut file which may or may not exist at the time the
237// drag is started. Therefore while waiting for that shortcut to be located we
238// just do a regular "internal" drag and transition into the synchronous drag
239// when the shortcut is found/created. Hence a synchronous drag is an optional
240// phase of a regular drag and non-Windows platforms drags are equivalent to a
241// Windows drag that never enters the synchronous drag phase.
242class SynchronousDrag : public ui::DragSourceWin {
243 public:
244  SynchronousDrag(AppsGridView* grid_view,
245                  AppListItemView* drag_view,
246                  const gfx::Point& drag_view_offset)
247      : grid_view_(grid_view),
248        drag_view_(drag_view),
249        drag_view_offset_(drag_view_offset),
250        has_shortcut_path_(false),
251        running_(false),
252        canceled_(false) {}
253
254  void set_shortcut_path(const base::FilePath& shortcut_path) {
255    has_shortcut_path_ = true;
256    shortcut_path_ = shortcut_path;
257  }
258
259  bool running() { return running_; }
260
261  bool CanRun() {
262    return has_shortcut_path_ && !running_;
263  }
264
265  void Run() {
266    DCHECK(CanRun());
267
268    // Prevent the synchronous dragger being destroyed while the drag is
269    // running.
270    scoped_refptr<SynchronousDrag> this_ref = this;
271    running_ = true;
272
273    ui::OSExchangeData data;
274    SetupExchangeData(&data);
275
276    // Hide the dragged view because the OS is going to create its own.
277    drag_view_->SetVisible(false);
278
279    // Blocks until the drag is finished. Calls into the ui::DragSourceWin
280    // methods.
281    DWORD effects;
282    DoDragDrop(ui::OSExchangeDataProviderWin::GetIDataObject(data),
283               this, DROPEFFECT_MOVE | DROPEFFECT_LINK, &effects);
284
285    // If |drag_view_| is NULL the drag was ended by some reentrant code.
286    if (drag_view_) {
287      // Make the drag view visible again.
288      drag_view_->SetVisible(true);
289      drag_view_->OnSyncDragEnd();
290
291      grid_view_->EndDrag(canceled_ || !IsCursorWithinGridView());
292    }
293  }
294
295  void EndDragExternally() {
296    CancelDrag();
297    DCHECK(drag_view_);
298    drag_view_->SetVisible(true);
299    drag_view_ = NULL;
300  }
301
302 private:
303  // Overridden from ui::DragSourceWin.
304  virtual void OnDragSourceCancel() OVERRIDE {
305    canceled_ = true;
306  }
307
308  virtual void OnDragSourceDrop() OVERRIDE {
309  }
310
311  virtual void OnDragSourceMove() OVERRIDE {
312    grid_view_->UpdateDrag(AppsGridView::MOUSE, GetCursorInGridViewCoords());
313  }
314
315  void SetupExchangeData(ui::OSExchangeData* data) {
316    data->SetFilename(shortcut_path_);
317    gfx::ImageSkia image(drag_view_->GetDragImage());
318    gfx::Size image_size(image.size());
319    drag_utils::SetDragImageOnDataObject(
320        image,
321        drag_view_offset_ - drag_view_->GetDragImageOffset(),
322        data);
323  }
324
325  HWND GetGridViewHWND() {
326    return views::HWNDForView(grid_view_);
327  }
328
329  bool IsCursorWithinGridView() {
330    POINT p;
331    GetCursorPos(&p);
332    return GetGridViewHWND() == WindowFromPoint(p);
333  }
334
335  gfx::Point GetCursorInGridViewCoords() {
336    POINT p;
337    GetCursorPos(&p);
338    ScreenToClient(GetGridViewHWND(), &p);
339    gfx::Point grid_view_pt(p.x, p.y);
340    grid_view_pt = gfx::win::ScreenToDIPPoint(grid_view_pt);
341    views::View::ConvertPointFromWidget(grid_view_, &grid_view_pt);
342    return grid_view_pt;
343  }
344
345  AppsGridView* grid_view_;
346  AppListItemView* drag_view_;
347  gfx::Point drag_view_offset_;
348  bool has_shortcut_path_;
349  base::FilePath shortcut_path_;
350  bool running_;
351  bool canceled_;
352
353  DISALLOW_COPY_AND_ASSIGN(SynchronousDrag);
354};
355#endif  // defined(OS_WIN)
356
357AppsGridView::AppsGridView(AppsGridViewDelegate* delegate)
358    : model_(NULL),
359      item_list_(NULL),
360      delegate_(delegate),
361      folder_delegate_(NULL),
362      page_switcher_view_(NULL),
363      cols_(0),
364      rows_per_page_(0),
365      selected_view_(NULL),
366      drag_view_(NULL),
367      drag_start_page_(-1),
368#if defined(OS_WIN)
369      use_synchronous_drag_(true),
370#endif
371      drag_pointer_(NONE),
372      drop_attempt_(DROP_FOR_NONE),
373      drag_and_drop_host_(NULL),
374      forward_events_to_drag_and_drop_host_(false),
375      page_flip_target_(-1),
376      page_flip_delay_in_ms_(kPageFlipDelayInMs),
377      bounds_animator_(this),
378      activated_folder_item_view_(NULL),
379      dragging_for_reparent_item_(false) {
380  SetPaintToLayer(true);
381  // Clip any icons that are outside the grid view's bounds. These icons would
382  // otherwise be visible to the user when the grid view is off screen.
383  layer()->SetMasksToBounds(true);
384  SetFillsBoundsOpaquely(false);
385
386  pagination_model_.SetTransitionDurations(kPageTransitionDurationInMs,
387                                           kOverscrollPageTransitionDurationMs);
388
389  pagination_model_.AddObserver(this);
390  // The experimental app list transitions vertically.
391  PaginationController::ScrollAxis scroll_axis =
392      app_list::switches::IsExperimentalAppListEnabled()
393          ? PaginationController::SCROLL_AXIS_VERTICAL
394          : PaginationController::SCROLL_AXIS_HORIZONTAL;
395  pagination_controller_.reset(
396      new PaginationController(&pagination_model_, scroll_axis));
397  if (!switches::IsExperimentalAppListEnabled()) {
398    page_switcher_view_ = new PageSwitcher(&pagination_model_);
399    AddChildView(page_switcher_view_);
400  }
401}
402
403AppsGridView::~AppsGridView() {
404  // Coming here |drag_view_| should already be canceled since otherwise the
405  // drag would disappear after the app list got animated away and closed,
406  // which would look odd.
407  DCHECK(!drag_view_);
408  if (drag_view_)
409    EndDrag(true);
410
411  if (model_)
412    model_->RemoveObserver(this);
413  pagination_model_.RemoveObserver(this);
414
415  if (item_list_)
416    item_list_->RemoveObserver(this);
417
418  // Make sure |page_switcher_view_| is deleted before |pagination_model_|.
419  view_model_.Clear();
420  RemoveAllChildViews(true);
421}
422
423void AppsGridView::SetLayout(int cols, int rows_per_page) {
424  cols_ = cols;
425  rows_per_page_ = rows_per_page;
426
427  SetBorder(views::Border::CreateEmptyBorder(
428      switches::IsExperimentalAppListEnabled() ? 0 : kTopPadding,
429      kLeftRightPadding,
430      0,
431      kLeftRightPadding));
432}
433
434void AppsGridView::ResetForShowApps() {
435  activated_folder_item_view_ = NULL;
436  ClearDragState();
437  layer()->SetOpacity(1.0f);
438  SetVisible(true);
439  // Set all views to visible in case they weren't made visible again by an
440  // incomplete animation.
441  for (int i = 0; i < view_model_.view_size(); ++i) {
442    view_model_.view_at(i)->SetVisible(true);
443  }
444  CHECK_EQ(item_list_->item_count(),
445           static_cast<size_t>(view_model_.view_size()));
446}
447
448void AppsGridView::SetModel(AppListModel* model) {
449  if (model_)
450    model_->RemoveObserver(this);
451
452  model_ = model;
453  if (model_)
454    model_->AddObserver(this);
455
456  Update();
457}
458
459void AppsGridView::SetItemList(AppListItemList* item_list) {
460  if (item_list_)
461    item_list_->RemoveObserver(this);
462  item_list_ = item_list;
463  if (item_list_)
464    item_list_->AddObserver(this);
465  Update();
466}
467
468void AppsGridView::SetSelectedView(views::View* view) {
469  if (IsSelectedView(view) || IsDraggedView(view))
470    return;
471
472  Index index = GetIndexOfView(view);
473  if (IsValidIndex(index))
474    SetSelectedItemByIndex(index);
475}
476
477void AppsGridView::ClearSelectedView(views::View* view) {
478  if (view && IsSelectedView(view)) {
479    selected_view_->SchedulePaint();
480    selected_view_ = NULL;
481  }
482}
483
484void AppsGridView::ClearAnySelectedView() {
485  if (selected_view_) {
486    selected_view_->SchedulePaint();
487    selected_view_ = NULL;
488  }
489}
490
491bool AppsGridView::IsSelectedView(const views::View* view) const {
492  return selected_view_ == view;
493}
494
495void AppsGridView::EnsureViewVisible(const views::View* view) {
496  if (pagination_model_.has_transition())
497    return;
498
499  Index index = GetIndexOfView(view);
500  if (IsValidIndex(index))
501    pagination_model_.SelectPage(index.page, false);
502}
503
504void AppsGridView::InitiateDrag(AppListItemView* view,
505                                Pointer pointer,
506                                const ui::LocatedEvent& event) {
507  DCHECK(view);
508  if (drag_view_ || pulsing_blocks_model_.view_size())
509    return;
510
511  drag_view_ = view;
512  drag_view_init_index_ = GetIndexOfView(drag_view_);
513  drag_view_offset_ = event.location();
514  drag_start_page_ = pagination_model_.selected_page();
515  reorder_placeholder_ = drag_view_init_index_;
516  ExtractDragLocation(event, &drag_start_grid_view_);
517  drag_view_start_ = gfx::Point(drag_view_->x(), drag_view_->y());
518}
519
520void AppsGridView::StartSettingUpSynchronousDrag() {
521#if defined(OS_WIN)
522  if (!delegate_ || !use_synchronous_drag_)
523    return;
524
525  // Folders and downloading items can't be integrated with the OS.
526  if (IsFolderItem(drag_view_->item()) || drag_view_->item()->is_installing())
527    return;
528
529  // Favor the drag and drop host over native win32 drag. For the Win8/ash
530  // launcher we want to have ashes drag and drop over win32's.
531  if (drag_and_drop_host_)
532    return;
533
534  // Never create a second synchronous drag if the drag started in a folder.
535  if (IsDraggingForReparentInRootLevelGridView())
536    return;
537
538  synchronous_drag_ = new SynchronousDrag(this, drag_view_, drag_view_offset_);
539  delegate_->GetShortcutPathForApp(drag_view_->item()->id(),
540                                   base::Bind(&AppsGridView::OnGotShortcutPath,
541                                              base::Unretained(this),
542                                              synchronous_drag_));
543#endif
544}
545
546bool AppsGridView::RunSynchronousDrag() {
547#if defined(OS_WIN)
548  if (!synchronous_drag_)
549    return false;
550
551  if (synchronous_drag_->CanRun()) {
552    if (IsDraggingForReparentInHiddenGridView())
553      folder_delegate_->SetRootLevelDragViewVisible(false);
554    synchronous_drag_->Run();
555    synchronous_drag_ = NULL;
556    return true;
557  } else if (!synchronous_drag_->running()) {
558    // The OS drag is not ready yet. If the root grid has a drag view because
559    // a reparent has started, ensure it is visible.
560    if (IsDraggingForReparentInHiddenGridView())
561      folder_delegate_->SetRootLevelDragViewVisible(true);
562  }
563#endif
564  return false;
565}
566
567void AppsGridView::CleanUpSynchronousDrag() {
568#if defined(OS_WIN)
569  if (synchronous_drag_)
570    synchronous_drag_->EndDragExternally();
571
572  synchronous_drag_ = NULL;
573#endif
574}
575
576#if defined(OS_WIN)
577void AppsGridView::OnGotShortcutPath(
578    scoped_refptr<SynchronousDrag> synchronous_drag,
579    const base::FilePath& path) {
580  // Drag may have ended before we get the shortcut path or a new drag may have
581  // begun.
582  if (synchronous_drag_ != synchronous_drag)
583    return;
584  // Setting the shortcut path here means the next time we hit UpdateDrag()
585  // we'll enter the synchronous drag.
586  // NOTE we don't Run() the drag here because that causes animations not to
587  // update for some reason.
588  synchronous_drag_->set_shortcut_path(path);
589  DCHECK(synchronous_drag_->CanRun());
590}
591#endif
592
593bool AppsGridView::UpdateDragFromItem(Pointer pointer,
594                                      const ui::LocatedEvent& event) {
595  if (!drag_view_)
596    return false;  // Drag canceled.
597
598  gfx::Point drag_point_in_grid_view;
599  ExtractDragLocation(event, &drag_point_in_grid_view);
600  UpdateDrag(pointer, drag_point_in_grid_view);
601  if (!dragging())
602    return false;
603
604  // If a drag and drop host is provided, see if the drag operation needs to be
605  // forwarded.
606  gfx::Point location_in_screen = drag_point_in_grid_view;
607  views::View::ConvertPointToScreen(this, &location_in_screen);
608  DispatchDragEventToDragAndDropHost(location_in_screen);
609  if (drag_and_drop_host_)
610    drag_and_drop_host_->UpdateDragIconProxy(location_in_screen);
611  return true;
612}
613
614void AppsGridView::UpdateDrag(Pointer pointer, const gfx::Point& point) {
615  if (folder_delegate_)
616    UpdateDragStateInsideFolder(pointer, point);
617
618  if (!drag_view_)
619    return;  // Drag canceled.
620
621  if (RunSynchronousDrag())
622    return;
623
624  gfx::Vector2d drag_vector(point - drag_start_grid_view_);
625  if (!dragging() && ExceededDragThreshold(drag_vector)) {
626    drag_pointer_ = pointer;
627    // Move the view to the front so that it appears on top of other views.
628    ReorderChildView(drag_view_, -1);
629    bounds_animator_.StopAnimatingView(drag_view_);
630    // Stopping the animation may have invalidated our drag view due to the
631    // view hierarchy changing.
632    if (!drag_view_)
633      return;
634
635    StartSettingUpSynchronousDrag();
636    if (!dragging_for_reparent_item_)
637      StartDragAndDropHostDrag(point);
638  }
639
640  if (drag_pointer_ != pointer)
641    return;
642
643  drag_view_->SetPosition(drag_view_start_ + drag_vector);
644
645  last_drag_point_ = point;
646  const Index last_reorder_drop_target = reorder_drop_target_;
647  const Index last_folder_drop_target = folder_drop_target_;
648  DropAttempt last_drop_attempt = drop_attempt_;
649  CalculateDropTarget();
650
651  MaybeStartPageFlipTimer(last_drag_point_);
652
653  if (page_switcher_view_) {
654    gfx::Point page_switcher_point(last_drag_point_);
655    views::View::ConvertPointToTarget(
656        this, page_switcher_view_, &page_switcher_point);
657    page_switcher_view_->UpdateUIForDragPoint(page_switcher_point);
658  }
659
660  if (last_folder_drop_target != folder_drop_target_ ||
661      last_reorder_drop_target != reorder_drop_target_ ||
662      last_drop_attempt != drop_attempt_) {
663    if (drop_attempt_ == DROP_FOR_REORDER) {
664      folder_dropping_timer_.Stop();
665      reorder_timer_.Start(FROM_HERE,
666          base::TimeDelta::FromMilliseconds(kReorderDelay),
667          this, &AppsGridView::OnReorderTimer);
668    } else if (drop_attempt_ == DROP_FOR_FOLDER) {
669      reorder_timer_.Stop();
670      folder_dropping_timer_.Start(FROM_HERE,
671          base::TimeDelta::FromMilliseconds(kFolderDroppingDelay),
672          this, &AppsGridView::OnFolderDroppingTimer);
673    }
674
675    // Reset the previous drop target.
676    SetAsFolderDroppingTarget(last_folder_drop_target, false);
677  }
678}
679
680void AppsGridView::EndDrag(bool cancel) {
681  // EndDrag was called before if |drag_view_| is NULL.
682  if (!drag_view_)
683    return;
684
685  // Coming here a drag and drop was in progress.
686  bool landed_in_drag_and_drop_host = forward_events_to_drag_and_drop_host_;
687  if (forward_events_to_drag_and_drop_host_) {
688    DCHECK(!IsDraggingForReparentInRootLevelGridView());
689    forward_events_to_drag_and_drop_host_ = false;
690    drag_and_drop_host_->EndDrag(cancel);
691    if (IsDraggingForReparentInHiddenGridView()) {
692      folder_delegate_->DispatchEndDragEventForReparent(
693          true /* events_forwarded_to_drag_drop_host */,
694          cancel /* cancel_drag */);
695    }
696  } else {
697    if (IsDraggingForReparentInHiddenGridView()) {
698      // Forward the EndDrag event to the root level grid view.
699      folder_delegate_->DispatchEndDragEventForReparent(
700          false /* events_forwarded_to_drag_drop_host */,
701          cancel /* cancel_drag */);
702      EndDragForReparentInHiddenFolderGridView();
703      return;
704    }
705
706    if (IsDraggingForReparentInRootLevelGridView()) {
707      // An EndDrag can be received during a reparent via a model change. This
708      // is always a cancel and needs to be forwarded to the folder.
709      DCHECK(cancel);
710      delegate_->CancelDragInActiveFolder();
711      return;
712    }
713
714    if (!cancel && dragging()) {
715      // Regular drag ending path, ie, not for reparenting.
716      CalculateDropTarget();
717      if (EnableFolderDragDropUI() && drop_attempt_ == DROP_FOR_FOLDER &&
718          IsValidIndex(folder_drop_target_)) {
719        MoveItemToFolder(drag_view_, folder_drop_target_);
720      } else if (IsValidIndex(reorder_drop_target_)) {
721        MoveItemInModel(drag_view_, reorder_drop_target_);
722      }
723    }
724  }
725
726  if (drag_and_drop_host_) {
727    // If we had a drag and drop proxy icon, we delete it and make the real
728    // item visible again.
729    drag_and_drop_host_->DestroyDragIconProxy();
730    if (landed_in_drag_and_drop_host) {
731      // Move the item directly to the target location, avoiding the "zip back"
732      // animation if the user was pinning it to the shelf.
733      int i = reorder_drop_target_.slot;
734      gfx::Rect bounds = view_model_.ideal_bounds(i);
735      drag_view_->SetBoundsRect(bounds);
736    }
737    // Fade in slowly if it landed in the shelf.
738    SetViewHidden(drag_view_,
739                  false /* show */,
740                  !landed_in_drag_and_drop_host /* animate */);
741  }
742
743  // The drag can be ended after the synchronous drag is created but before it
744  // is Run().
745  CleanUpSynchronousDrag();
746
747  SetAsFolderDroppingTarget(folder_drop_target_, false);
748  ClearDragState();
749  AnimateToIdealBounds();
750
751  StopPageFlipTimer();
752
753  // If user releases mouse inside a folder's grid view, burst the folder
754  // container ink bubble.
755  if (folder_delegate_ && !IsDraggingForReparentInHiddenGridView())
756    folder_delegate_->UpdateFolderViewBackground(false);
757}
758
759void AppsGridView::StopPageFlipTimer() {
760  page_flip_timer_.Stop();
761  page_flip_target_ = -1;
762}
763
764AppListItemView* AppsGridView::GetItemViewAt(int index) const {
765  DCHECK(index >= 0 && index < view_model_.view_size());
766  return static_cast<AppListItemView*>(view_model_.view_at(index));
767}
768
769void AppsGridView::SetTopItemViewsVisible(bool visible) {
770  int top_item_count = std::min(static_cast<int>(kNumFolderTopItems),
771                                view_model_.view_size());
772  for (int i = 0; i < top_item_count; ++i)
773    GetItemViewAt(i)->icon()->SetVisible(visible);
774}
775
776void AppsGridView::ScheduleShowHideAnimation(bool show) {
777  // Stop any previous animation.
778  layer()->GetAnimator()->StopAnimating();
779
780  // Set initial state.
781  SetVisible(true);
782  layer()->SetOpacity(show ? 0.0f : 1.0f);
783
784  ui::ScopedLayerAnimationSettings animation(layer()->GetAnimator());
785  animation.AddObserver(this);
786  animation.SetTweenType(
787      show ? kFolderFadeInTweenType : kFolderFadeOutTweenType);
788  animation.SetTransitionDuration(base::TimeDelta::FromMilliseconds(
789      show ? kFolderTransitionInDurationMs : kFolderTransitionOutDurationMs));
790
791  layer()->SetOpacity(show ? 1.0f : 0.0f);
792}
793
794void AppsGridView::InitiateDragFromReparentItemInRootLevelGridView(
795    AppListItemView* original_drag_view,
796    const gfx::Rect& drag_view_rect,
797    const gfx::Point& drag_point) {
798  DCHECK(original_drag_view && !drag_view_);
799  DCHECK(!dragging_for_reparent_item_);
800
801  // Since the item is new, its placeholder is conceptually at the back of the
802  // entire apps grid.
803  reorder_placeholder_ = GetLastViewIndex();
804
805  // Create a new AppListItemView to duplicate the original_drag_view in the
806  // folder's grid view.
807  AppListItemView* view = new AppListItemView(this, original_drag_view->item());
808  AddChildView(view);
809  drag_view_ = view;
810  drag_view_->SetPaintToLayer(true);
811  // Note: For testing purpose, SetFillsBoundsOpaquely can be set to true to
812  // show the gray background.
813  drag_view_->SetFillsBoundsOpaquely(false);
814  drag_view_->SetBoundsRect(drag_view_rect);
815  drag_view_->SetDragUIState();  // Hide the title of the drag_view_.
816
817  // Hide the drag_view_ for drag icon proxy.
818  SetViewHidden(drag_view_,
819                true /* hide */,
820                true /* no animate */);
821
822  // Add drag_view_ to the end of the view_model_.
823  view_model_.Add(drag_view_, view_model_.view_size());
824
825  drag_start_page_ = pagination_model_.selected_page();
826  drag_start_grid_view_ = drag_point;
827
828  drag_view_start_ = gfx::Point(drag_view_->x(), drag_view_->y());
829
830  // Set the flag in root level grid view.
831  dragging_for_reparent_item_ = true;
832}
833
834void AppsGridView::UpdateDragFromReparentItem(Pointer pointer,
835                                              const gfx::Point& drag_point) {
836  // Note that if a cancel ocurrs while reparenting, the |drag_view_| in both
837  // root and folder grid views is cleared, so the check in UpdateDragFromItem()
838  // for |drag_view_| being NULL (in the folder grid) is sufficient.
839  DCHECK(drag_view_);
840  DCHECK(IsDraggingForReparentInRootLevelGridView());
841
842  UpdateDrag(pointer, drag_point);
843}
844
845bool AppsGridView::IsDraggedView(const views::View* view) const {
846  return drag_view_ == view;
847}
848
849void AppsGridView::ClearDragState() {
850  drop_attempt_ = DROP_FOR_NONE;
851  drag_pointer_ = NONE;
852  reorder_drop_target_ = Index();
853  folder_drop_target_ = Index();
854  reorder_placeholder_ = Index();
855  drag_start_grid_view_ = gfx::Point();
856  drag_start_page_ = -1;
857  drag_view_offset_ = gfx::Point();
858
859  if (drag_view_) {
860    drag_view_->OnDragEnded();
861    if (IsDraggingForReparentInRootLevelGridView()) {
862      const int drag_view_index = view_model_.GetIndexOfView(drag_view_);
863      CHECK_EQ(view_model_.view_size() - 1, drag_view_index);
864      DeleteItemViewAtIndex(drag_view_index);
865    }
866  }
867  drag_view_ = NULL;
868  dragging_for_reparent_item_ = false;
869}
870
871void AppsGridView::SetDragViewVisible(bool visible) {
872  DCHECK(drag_view_);
873  SetViewHidden(drag_view_, !visible, true);
874}
875
876void AppsGridView::SetDragAndDropHostOfCurrentAppList(
877    ApplicationDragAndDropHost* drag_and_drop_host) {
878  drag_and_drop_host_ = drag_and_drop_host;
879}
880
881void AppsGridView::Prerender() {
882  Layout();
883  int selected_page = std::max(0, pagination_model_.selected_page());
884  int start = std::max(0, (selected_page - kPrerenderPages) * tiles_per_page());
885  int end = std::min(view_model_.view_size(),
886                     (selected_page + 1 + kPrerenderPages) * tiles_per_page());
887  for (int i = start; i < end; i++) {
888    AppListItemView* v = static_cast<AppListItemView*>(view_model_.view_at(i));
889    v->Prerender();
890  }
891}
892
893bool AppsGridView::IsAnimatingView(views::View* view) {
894  return bounds_animator_.IsAnimating(view);
895}
896
897gfx::Size AppsGridView::GetPreferredSize() const {
898  const gfx::Insets insets(GetInsets());
899  int page_switcher_height = 0;
900  if (page_switcher_view_)
901    page_switcher_height = page_switcher_view_->GetPreferredSize().height();
902  gfx::Size size = GetTileGridSize();
903  size.Enlarge(insets.width(), insets.height() + page_switcher_height);
904  return size;
905}
906
907bool AppsGridView::GetDropFormats(
908    int* formats,
909    std::set<OSExchangeData::CustomFormat>* custom_formats) {
910  // TODO(koz): Only accept a specific drag type for app shortcuts.
911  *formats = OSExchangeData::FILE_NAME;
912  return true;
913}
914
915bool AppsGridView::CanDrop(const OSExchangeData& data) {
916  return true;
917}
918
919int AppsGridView::OnDragUpdated(const ui::DropTargetEvent& event) {
920  return ui::DragDropTypes::DRAG_MOVE;
921}
922
923void AppsGridView::Layout() {
924  if (bounds_animator_.IsAnimating())
925    bounds_animator_.Cancel();
926
927  CalculateIdealBounds();
928  for (int i = 0; i < view_model_.view_size(); ++i) {
929    views::View* view = view_model_.view_at(i);
930    if (view != drag_view_)
931      view->SetBoundsRect(view_model_.ideal_bounds(i));
932  }
933  views::ViewModelUtils::SetViewBoundsToIdealBounds(pulsing_blocks_model_);
934
935  if (page_switcher_view_) {
936    const int page_switcher_height =
937        page_switcher_view_->GetPreferredSize().height();
938    gfx::Rect rect(GetContentsBounds());
939    rect.set_y(rect.bottom() - page_switcher_height);
940    rect.set_height(page_switcher_height);
941    page_switcher_view_->SetBoundsRect(rect);
942  }
943}
944
945bool AppsGridView::OnKeyPressed(const ui::KeyEvent& event) {
946  bool handled = false;
947  if (selected_view_)
948    handled = selected_view_->OnKeyPressed(event);
949
950  if (!handled) {
951    const int forward_dir = base::i18n::IsRTL() ? -1 : 1;
952    switch (event.key_code()) {
953      case ui::VKEY_LEFT:
954        MoveSelected(0, -forward_dir, 0);
955        return true;
956      case ui::VKEY_RIGHT:
957        MoveSelected(0, forward_dir, 0);
958        return true;
959      case ui::VKEY_UP:
960        MoveSelected(0, 0, -1);
961        return true;
962      case ui::VKEY_DOWN:
963        MoveSelected(0, 0, 1);
964        return true;
965      case ui::VKEY_PRIOR: {
966        MoveSelected(-1, 0, 0);
967        return true;
968      }
969      case ui::VKEY_NEXT: {
970        MoveSelected(1, 0, 0);
971        return true;
972      }
973      default:
974        break;
975    }
976  }
977
978  return handled;
979}
980
981bool AppsGridView::OnKeyReleased(const ui::KeyEvent& event) {
982  bool handled = false;
983  if (selected_view_)
984    handled = selected_view_->OnKeyReleased(event);
985
986  return handled;
987}
988
989bool AppsGridView::OnMouseWheel(const ui::MouseWheelEvent& event) {
990  return pagination_controller_->OnScroll(
991      gfx::Vector2d(event.x_offset(), event.y_offset()));
992}
993
994void AppsGridView::ViewHierarchyChanged(
995    const ViewHierarchyChangedDetails& details) {
996  if (!details.is_add && details.parent == this) {
997    // The view being delete should not have reference in |view_model_|.
998    CHECK_EQ(-1, view_model_.GetIndexOfView(details.child));
999
1000    if (selected_view_ == details.child)
1001      selected_view_ = NULL;
1002    if (activated_folder_item_view_ == details.child)
1003      activated_folder_item_view_ = NULL;
1004
1005    if (drag_view_ == details.child)
1006      EndDrag(true);
1007
1008    bounds_animator_.StopAnimatingView(details.child);
1009  }
1010}
1011
1012void AppsGridView::OnGestureEvent(ui::GestureEvent* event) {
1013  if (pagination_controller_->OnGestureEvent(*event, GetContentsBounds()))
1014    event->SetHandled();
1015}
1016
1017void AppsGridView::OnScrollEvent(ui::ScrollEvent* event) {
1018  if (event->type() == ui::ET_SCROLL_FLING_CANCEL)
1019    return;
1020
1021  gfx::Vector2dF offset(event->x_offset(), event->y_offset());
1022  if (pagination_controller_->OnScroll(gfx::ToFlooredVector2d(offset))) {
1023    event->SetHandled();
1024    event->StopPropagation();
1025  }
1026}
1027
1028void AppsGridView::Update() {
1029  DCHECK(!selected_view_ && !drag_view_);
1030  view_model_.Clear();
1031  if (!item_list_ || !item_list_->item_count())
1032    return;
1033  for (size_t i = 0; i < item_list_->item_count(); ++i) {
1034    views::View* view = CreateViewForItemAtIndex(i);
1035    view_model_.Add(view, i);
1036    AddChildView(view);
1037  }
1038  UpdatePaging();
1039  UpdatePulsingBlockViews();
1040  Layout();
1041  SchedulePaint();
1042}
1043
1044void AppsGridView::UpdatePaging() {
1045  int total_page = view_model_.view_size() && tiles_per_page()
1046                       ? (view_model_.view_size() - 1) / tiles_per_page() + 1
1047                       : 0;
1048
1049  pagination_model_.SetTotalPages(total_page);
1050}
1051
1052void AppsGridView::UpdatePulsingBlockViews() {
1053  const int existing_items = item_list_ ? item_list_->item_count() : 0;
1054  const int available_slots =
1055      tiles_per_page() - existing_items % tiles_per_page();
1056  const int desired = model_->status() == AppListModel::STATUS_SYNCING ?
1057      available_slots : 0;
1058
1059  if (pulsing_blocks_model_.view_size() == desired)
1060    return;
1061
1062  while (pulsing_blocks_model_.view_size() > desired) {
1063    views::View* view = pulsing_blocks_model_.view_at(0);
1064    pulsing_blocks_model_.Remove(0);
1065    delete view;
1066  }
1067
1068  while (pulsing_blocks_model_.view_size() < desired) {
1069    views::View* view = new PulsingBlockView(GetTotalTileSize(), true);
1070    pulsing_blocks_model_.Add(view, 0);
1071    AddChildView(view);
1072  }
1073}
1074
1075views::View* AppsGridView::CreateViewForItemAtIndex(size_t index) {
1076  // The drag_view_ might be pending for deletion, therefore view_model_
1077  // may have one more item than item_list_.
1078  DCHECK_LE(index, item_list_->item_count());
1079  AppListItemView* view = new AppListItemView(this,
1080                                              item_list_->item_at(index));
1081  view->SetPaintToLayer(true);
1082  view->SetFillsBoundsOpaquely(false);
1083  return view;
1084}
1085
1086AppsGridView::Index AppsGridView::GetIndexFromModelIndex(
1087    int model_index) const {
1088  return Index(model_index / tiles_per_page(), model_index % tiles_per_page());
1089}
1090
1091int AppsGridView::GetModelIndexFromIndex(const Index& index) const {
1092  return index.page * tiles_per_page() + index.slot;
1093}
1094
1095void AppsGridView::SetSelectedItemByIndex(const Index& index) {
1096  if (GetIndexOfView(selected_view_) == index)
1097    return;
1098
1099  views::View* new_selection = GetViewAtIndex(index);
1100  if (!new_selection)
1101    return;  // Keep current selection.
1102
1103  if (selected_view_)
1104    selected_view_->SchedulePaint();
1105
1106  EnsureViewVisible(new_selection);
1107  selected_view_ = new_selection;
1108  selected_view_->SchedulePaint();
1109  selected_view_->NotifyAccessibilityEvent(
1110      ui::AX_EVENT_FOCUS, true);
1111}
1112
1113bool AppsGridView::IsValidIndex(const Index& index) const {
1114  return index.page >= 0 && index.page < pagination_model_.total_pages() &&
1115         index.slot >= 0 && index.slot < tiles_per_page() &&
1116         GetModelIndexFromIndex(index) < view_model_.view_size();
1117}
1118
1119AppsGridView::Index AppsGridView::GetIndexOfView(
1120    const views::View* view) const {
1121  const int model_index = view_model_.GetIndexOfView(view);
1122  if (model_index == -1)
1123    return Index();
1124
1125  return GetIndexFromModelIndex(model_index);
1126}
1127
1128views::View* AppsGridView::GetViewAtIndex(const Index& index) const {
1129  if (!IsValidIndex(index))
1130    return NULL;
1131
1132  const int model_index = GetModelIndexFromIndex(index);
1133  return view_model_.view_at(model_index);
1134}
1135
1136AppsGridView::Index AppsGridView::GetLastViewIndex() const {
1137  DCHECK_LT(0, view_model_.view_size());
1138  int view_index = view_model_.view_size() - 1;
1139  return Index(view_index / tiles_per_page(), view_index % tiles_per_page());
1140}
1141
1142void AppsGridView::MoveSelected(int page_delta,
1143                                int slot_x_delta,
1144                                int slot_y_delta) {
1145  if (!selected_view_)
1146    return SetSelectedItemByIndex(Index(pagination_model_.selected_page(), 0));
1147
1148  const Index& selected = GetIndexOfView(selected_view_);
1149  int target_slot = selected.slot + slot_x_delta + slot_y_delta * cols_;
1150
1151  if (selected.slot % cols_ == 0 && slot_x_delta == -1) {
1152    if (selected.page > 0) {
1153      page_delta = -1;
1154      target_slot = selected.slot + cols_ - 1;
1155    } else {
1156      target_slot = selected.slot;
1157    }
1158  }
1159
1160  if (selected.slot % cols_ == cols_ - 1 && slot_x_delta == 1) {
1161    if (selected.page < pagination_model_.total_pages() - 1) {
1162      page_delta = 1;
1163      target_slot = selected.slot - cols_ + 1;
1164    } else {
1165      target_slot = selected.slot;
1166    }
1167  }
1168
1169  // Clamp the target slot to the last item if we are moving to the last page
1170  // but our target slot is past the end of the item list.
1171  if (page_delta &&
1172      selected.page + page_delta == pagination_model_.total_pages() - 1) {
1173    int last_item_slot = (view_model_.view_size() - 1) % tiles_per_page();
1174    if (last_item_slot < target_slot) {
1175      target_slot = last_item_slot;
1176    }
1177  }
1178
1179  int target_page = std::min(pagination_model_.total_pages() - 1,
1180                             std::max(selected.page + page_delta, 0));
1181  SetSelectedItemByIndex(Index(target_page, target_slot));
1182}
1183
1184void AppsGridView::CalculateIdealBounds() {
1185  // TODO(calamity): This fixes http://crbug.com/422604 on ChromeOS but it's
1186  // unclear why. This should be investigated to fix the issue on Linux Ash.
1187  if (GetContentsBounds().IsEmpty())
1188    return;
1189
1190  gfx::Size grid_size = GetTileGridSize();
1191
1192  // Page size including padding pixels. A tile.x + page_width means the same
1193  // tile slot in the next page; similarly for tile.y + page_height.
1194  const int page_width = grid_size.width() + kPagePadding;
1195  const int page_height = grid_size.height() + kPagePadding;
1196
1197  // If there is a transition, calculates offset for current and target page.
1198  const int current_page = pagination_model_.selected_page();
1199  const PaginationModel::Transition& transition =
1200      pagination_model_.transition();
1201  const bool is_valid = pagination_model_.is_valid_page(transition.target_page);
1202
1203  // Transition to previous page means negative offset.
1204  const int dir = transition.target_page > current_page ? -1 : 1;
1205
1206  const int total_views =
1207      view_model_.view_size() + pulsing_blocks_model_.view_size();
1208  int slot_index = 0;
1209  for (int i = 0; i < total_views; ++i) {
1210    if (i < view_model_.view_size() && view_model_.view_at(i) == drag_view_)
1211      continue;
1212
1213    Index view_index = GetIndexFromModelIndex(slot_index);
1214
1215    // Leaves a blank space in the grid for the current reorder placeholder.
1216    if (reorder_placeholder_ == view_index) {
1217      ++slot_index;
1218      view_index = GetIndexFromModelIndex(slot_index);
1219    }
1220
1221    // Decide the x or y offset for current item.
1222    int x_offset = 0;
1223    int y_offset = 0;
1224
1225    if (pagination_controller_->scroll_axis() ==
1226        PaginationController::SCROLL_AXIS_HORIZONTAL) {
1227      if (view_index.page < current_page)
1228        x_offset = -page_width;
1229      else if (view_index.page > current_page)
1230        x_offset = page_width;
1231
1232      if (is_valid) {
1233        if (view_index.page == current_page ||
1234            view_index.page == transition.target_page) {
1235          x_offset += transition.progress * page_width * dir;
1236        }
1237      }
1238    } else {
1239      if (view_index.page < current_page)
1240        y_offset = -page_height;
1241      else if (view_index.page > current_page)
1242        y_offset = page_height;
1243
1244      if (is_valid) {
1245        if (view_index.page == current_page ||
1246            view_index.page == transition.target_page) {
1247          y_offset += transition.progress * page_height * dir;
1248        }
1249      }
1250    }
1251
1252    const int row = view_index.slot / cols_;
1253    const int col = view_index.slot % cols_;
1254    gfx::Rect tile_slot = GetExpectedTileBounds(row, col);
1255    tile_slot.Offset(x_offset, y_offset);
1256    if (i < view_model_.view_size()) {
1257      view_model_.set_ideal_bounds(i, tile_slot);
1258    } else {
1259      pulsing_blocks_model_.set_ideal_bounds(i - view_model_.view_size(),
1260                                             tile_slot);
1261    }
1262
1263    ++slot_index;
1264  }
1265}
1266
1267void AppsGridView::AnimateToIdealBounds() {
1268  const gfx::Rect visible_bounds(GetVisibleBounds());
1269
1270  CalculateIdealBounds();
1271  for (int i = 0; i < view_model_.view_size(); ++i) {
1272    views::View* view = view_model_.view_at(i);
1273    if (view == drag_view_)
1274      continue;
1275
1276    const gfx::Rect& target = view_model_.ideal_bounds(i);
1277    if (bounds_animator_.GetTargetBounds(view) == target)
1278      continue;
1279
1280    const gfx::Rect& current = view->bounds();
1281    const bool current_visible = visible_bounds.Intersects(current);
1282    const bool target_visible = visible_bounds.Intersects(target);
1283    const bool visible = current_visible || target_visible;
1284
1285    const int y_diff = target.y() - current.y();
1286    if (visible && y_diff && y_diff % GetTotalTileSize().height() == 0) {
1287      AnimationBetweenRows(view,
1288                           current_visible,
1289                           current,
1290                           target_visible,
1291                           target);
1292    } else if (visible || bounds_animator_.IsAnimating(view)) {
1293      bounds_animator_.AnimateViewTo(view, target);
1294      bounds_animator_.SetAnimationDelegate(
1295          view,
1296          scoped_ptr<gfx::AnimationDelegate>(
1297              new ItemMoveAnimationDelegate(view)));
1298    } else {
1299      view->SetBoundsRect(target);
1300    }
1301  }
1302}
1303
1304void AppsGridView::AnimationBetweenRows(views::View* view,
1305                                        bool animate_current,
1306                                        const gfx::Rect& current,
1307                                        bool animate_target,
1308                                        const gfx::Rect& target) {
1309  // Determine page of |current| and |target|. -1 means in the left invisible
1310  // page, 0 is the center visible page and 1 means in the right invisible page.
1311  const int current_page = current.x() < 0 ? -1 :
1312      current.x() >= width() ? 1 : 0;
1313  const int target_page = target.x() < 0 ? -1 :
1314      target.x() >= width() ? 1 : 0;
1315
1316  const int dir = current_page < target_page ||
1317      (current_page == target_page && current.y() < target.y()) ? 1 : -1;
1318
1319  scoped_ptr<ui::Layer> layer;
1320  if (animate_current) {
1321    layer = view->RecreateLayer();
1322    layer->SuppressPaint();
1323
1324    view->SetFillsBoundsOpaquely(false);
1325    view->layer()->SetOpacity(0.f);
1326  }
1327
1328  gfx::Size total_tile_size = GetTotalTileSize();
1329  gfx::Rect current_out(current);
1330  current_out.Offset(dir * total_tile_size.width(), 0);
1331
1332  gfx::Rect target_in(target);
1333  if (animate_target)
1334    target_in.Offset(-dir * total_tile_size.width(), 0);
1335  view->SetBoundsRect(target_in);
1336  bounds_animator_.AnimateViewTo(view, target);
1337
1338  bounds_animator_.SetAnimationDelegate(
1339      view,
1340      scoped_ptr<gfx::AnimationDelegate>(
1341          new RowMoveAnimationDelegate(view, layer.release(), current_out)));
1342}
1343
1344void AppsGridView::ExtractDragLocation(const ui::LocatedEvent& event,
1345                                       gfx::Point* drag_point) {
1346#if defined(USE_AURA) && !defined(OS_WIN)
1347  // Use root location of |event| instead of location in |drag_view_|'s
1348  // coordinates because |drag_view_| has a scale transform and location
1349  // could have integer round error and causes jitter.
1350  *drag_point = event.root_location();
1351
1352  // GetWidget() could be NULL for tests.
1353  if (GetWidget()) {
1354    aura::Window::ConvertPointToTarget(
1355        GetWidget()->GetNativeWindow()->GetRootWindow(),
1356        GetWidget()->GetNativeWindow(),
1357        drag_point);
1358  }
1359
1360  views::View::ConvertPointFromWidget(this, drag_point);
1361#else
1362  // For non-aura, root location is not clearly defined but |drag_view_| does
1363  // not have the scale transform. So no round error would be introduced and
1364  // it's okay to use View::ConvertPointToTarget.
1365  *drag_point = event.location();
1366  views::View::ConvertPointToTarget(drag_view_, this, drag_point);
1367#endif
1368}
1369
1370void AppsGridView::CalculateDropTarget() {
1371  DCHECK(drag_view_);
1372
1373  gfx::Point point = drag_view_->icon()->bounds().CenterPoint();
1374  views::View::ConvertPointToTarget(drag_view_, this, &point);
1375  if (!IsPointWithinDragBuffer(point)) {
1376    // Reset the reorder target to the original position if the cursor is
1377    // outside the drag buffer.
1378    if (IsDraggingForReparentInRootLevelGridView()) {
1379      drop_attempt_ = DROP_FOR_NONE;
1380      return;
1381    }
1382
1383    reorder_drop_target_ = drag_view_init_index_;
1384    drop_attempt_ = DROP_FOR_REORDER;
1385    return;
1386  }
1387
1388  if (EnableFolderDragDropUI() &&
1389      CalculateFolderDropTarget(point, &folder_drop_target_)) {
1390    drop_attempt_ = DROP_FOR_FOLDER;
1391    return;
1392  }
1393
1394  drop_attempt_ = DROP_FOR_REORDER;
1395  CalculateReorderDropTarget(point, &reorder_drop_target_);
1396}
1397
1398bool AppsGridView::CalculateFolderDropTarget(const gfx::Point& point,
1399                                             Index* drop_target) const {
1400  Index nearest_tile_index(GetNearestTileIndexForPoint(point));
1401  int distance_to_tile_center =
1402      (point - GetExpectedTileBounds(nearest_tile_index.slot).CenterPoint())
1403          .Length();
1404  if (nearest_tile_index != reorder_placeholder_ &&
1405      distance_to_tile_center < kFolderDroppingCircleRadius &&
1406      !IsFolderItem(drag_view_->item()) &&
1407      CanDropIntoTarget(nearest_tile_index)) {
1408    *drop_target = nearest_tile_index;
1409    DCHECK(IsValidIndex(*drop_target));
1410    return true;
1411  }
1412
1413  return false;
1414}
1415
1416void AppsGridView::CalculateReorderDropTarget(const gfx::Point& point,
1417                                              Index* drop_target) const {
1418  gfx::Rect bounds = GetContentsBounds();
1419  Index grid_index = GetNearestTileIndexForPoint(point);
1420  gfx::Point reorder_placeholder_center =
1421      GetExpectedTileBounds(reorder_placeholder_.slot).CenterPoint();
1422
1423  int x_offset_direction = 0;
1424  if (grid_index == reorder_placeholder_) {
1425    x_offset_direction = reorder_placeholder_center.x() < point.x() ? -1 : 1;
1426  } else {
1427    x_offset_direction = reorder_placeholder_ < grid_index ? -1 : 1;
1428  }
1429
1430  gfx::Size total_tile_size = GetTotalTileSize();
1431  int row = grid_index.slot / cols_;
1432
1433  // Offset the target column based on the direction of the target. This will
1434  // result in earlier targets getting their reorder zone shifted backwards
1435  // and later targets getting their reorder zones shifted forwards.
1436  //
1437  // This makes eordering feel like the user is slotting items into the spaces
1438  // between apps.
1439  int x_offset = x_offset_direction *
1440                 (total_tile_size.width() - kFolderDroppingCircleRadius) / 2;
1441  int col = (point.x() - bounds.x() + x_offset) / total_tile_size.width();
1442  col = ClampToRange(col, 0, cols_ - 1);
1443  *drop_target =
1444      std::min(Index(pagination_model_.selected_page(), row * cols_ + col),
1445               GetLastViewIndex());
1446  DCHECK(IsValidIndex(*drop_target));
1447}
1448
1449void AppsGridView::OnReorderTimer() {
1450  if (drop_attempt_ == DROP_FOR_REORDER) {
1451    reorder_placeholder_ = reorder_drop_target_;
1452    AnimateToIdealBounds();
1453  }
1454}
1455
1456void AppsGridView::OnFolderItemReparentTimer() {
1457  DCHECK(folder_delegate_);
1458  if (drag_out_of_folder_container_ && drag_view_) {
1459    folder_delegate_->ReparentItem(drag_view_, last_drag_point_);
1460
1461    // Set the flag in the folder's grid view.
1462    dragging_for_reparent_item_ = true;
1463
1464    // Do not observe any data change since it is going to be hidden.
1465    item_list_->RemoveObserver(this);
1466    item_list_ = NULL;
1467  }
1468}
1469
1470void AppsGridView::OnFolderDroppingTimer() {
1471  if (drop_attempt_ == DROP_FOR_FOLDER)
1472    SetAsFolderDroppingTarget(folder_drop_target_, true);
1473}
1474
1475void AppsGridView::UpdateDragStateInsideFolder(Pointer pointer,
1476                                               const gfx::Point& drag_point) {
1477  if (IsUnderOEMFolder())
1478    return;
1479
1480  if (IsDraggingForReparentInHiddenGridView()) {
1481    // Dispatch drag event to root level grid view for re-parenting folder
1482    // folder item purpose.
1483    DispatchDragEventForReparent(pointer, drag_point);
1484    return;
1485  }
1486
1487  // Regular drag and drop in a folder's grid view.
1488  folder_delegate_->UpdateFolderViewBackground(true);
1489
1490  // Calculate if the drag_view_ is dragged out of the folder's container
1491  // ink bubble.
1492  gfx::Rect bounds_to_folder_view = ConvertRectToParent(drag_view_->bounds());
1493  gfx::Point pt = bounds_to_folder_view.CenterPoint();
1494  bool is_item_dragged_out_of_folder =
1495      folder_delegate_->IsPointOutsideOfFolderBoundary(pt);
1496  if (is_item_dragged_out_of_folder) {
1497    if (!drag_out_of_folder_container_) {
1498      folder_item_reparent_timer_.Start(
1499          FROM_HERE,
1500          base::TimeDelta::FromMilliseconds(kFolderItemReparentDelay),
1501          this,
1502          &AppsGridView::OnFolderItemReparentTimer);
1503      drag_out_of_folder_container_ = true;
1504    }
1505  } else {
1506    folder_item_reparent_timer_.Stop();
1507    drag_out_of_folder_container_ = false;
1508  }
1509}
1510
1511bool AppsGridView::IsDraggingForReparentInRootLevelGridView() const {
1512  return (!folder_delegate_ && dragging_for_reparent_item_);
1513}
1514
1515bool AppsGridView::IsDraggingForReparentInHiddenGridView() const {
1516  return (folder_delegate_ && dragging_for_reparent_item_);
1517}
1518
1519gfx::Rect AppsGridView::GetTargetIconRectInFolder(
1520    AppListItemView* drag_item_view,
1521    AppListItemView* folder_item_view) {
1522  gfx::Rect view_ideal_bounds = view_model_.ideal_bounds(
1523      view_model_.GetIndexOfView(folder_item_view));
1524  gfx::Rect icon_ideal_bounds =
1525      folder_item_view->GetIconBoundsForTargetViewBounds(view_ideal_bounds);
1526  AppListFolderItem* folder_item =
1527      static_cast<AppListFolderItem*>(folder_item_view->item());
1528  return folder_item->GetTargetIconRectInFolderForItem(
1529      drag_item_view->item(), icon_ideal_bounds);
1530}
1531
1532bool AppsGridView::IsUnderOEMFolder() {
1533  if (!folder_delegate_)
1534    return false;
1535
1536  return folder_delegate_->IsOEMFolder();
1537}
1538
1539void AppsGridView::DispatchDragEventForReparent(Pointer pointer,
1540                                                const gfx::Point& drag_point) {
1541  folder_delegate_->DispatchDragEventForReparent(pointer, drag_point);
1542}
1543
1544void AppsGridView::EndDragFromReparentItemInRootLevel(
1545    bool events_forwarded_to_drag_drop_host,
1546    bool cancel_drag) {
1547  // EndDrag was called before if |drag_view_| is NULL.
1548  if (!drag_view_)
1549    return;
1550
1551  DCHECK(IsDraggingForReparentInRootLevelGridView());
1552  bool cancel_reparent = cancel_drag || drop_attempt_ == DROP_FOR_NONE;
1553  if (!events_forwarded_to_drag_drop_host && !cancel_reparent) {
1554    CalculateDropTarget();
1555    if (drop_attempt_ == DROP_FOR_REORDER &&
1556        IsValidIndex(reorder_drop_target_)) {
1557      ReparentItemForReorder(drag_view_, reorder_drop_target_);
1558    } else if (drop_attempt_ == DROP_FOR_FOLDER &&
1559               IsValidIndex(folder_drop_target_)) {
1560      ReparentItemToAnotherFolder(drag_view_, folder_drop_target_);
1561    } else {
1562      NOTREACHED();
1563    }
1564    SetViewHidden(drag_view_, false /* show */, true /* no animate */);
1565  }
1566
1567  // The drag can be ended after the synchronous drag is created but before it
1568  // is Run().
1569  CleanUpSynchronousDrag();
1570
1571  SetAsFolderDroppingTarget(folder_drop_target_, false);
1572  if (cancel_reparent) {
1573    CancelFolderItemReparent(drag_view_);
1574  } else {
1575    // By setting |drag_view_| to NULL here, we prevent ClearDragState() from
1576    // cleaning up the newly created AppListItemView, effectively claiming
1577    // ownership of the newly created drag view.
1578    drag_view_->OnDragEnded();
1579    drag_view_ = NULL;
1580  }
1581  ClearDragState();
1582  AnimateToIdealBounds();
1583
1584  StopPageFlipTimer();
1585}
1586
1587void AppsGridView::EndDragForReparentInHiddenFolderGridView() {
1588  if (drag_and_drop_host_) {
1589    // If we had a drag and drop proxy icon, we delete it and make the real
1590    // item visible again.
1591    drag_and_drop_host_->DestroyDragIconProxy();
1592  }
1593
1594  // The drag can be ended after the synchronous drag is created but before it
1595  // is Run().
1596  CleanUpSynchronousDrag();
1597
1598  SetAsFolderDroppingTarget(folder_drop_target_, false);
1599  ClearDragState();
1600}
1601
1602void AppsGridView::OnFolderItemRemoved() {
1603  DCHECK(folder_delegate_);
1604  item_list_ = NULL;
1605}
1606
1607void AppsGridView::StartDragAndDropHostDrag(const gfx::Point& grid_location) {
1608  // When a drag and drop host is given, the item can be dragged out of the app
1609  // list window. In that case a proxy widget needs to be used.
1610  // Note: This code has very likely to be changed for Windows (non metro mode)
1611  // when a |drag_and_drop_host_| gets implemented.
1612  if (!drag_view_ || !drag_and_drop_host_)
1613    return;
1614
1615  gfx::Point screen_location = grid_location;
1616  views::View::ConvertPointToScreen(this, &screen_location);
1617
1618  // Determine the mouse offset to the center of the icon so that the drag and
1619  // drop host follows this layer.
1620  gfx::Vector2d delta = drag_view_offset_ -
1621                        drag_view_->GetLocalBounds().CenterPoint();
1622  delta.set_y(delta.y() + drag_view_->title()->size().height() / 2);
1623
1624  // We have to hide the original item since the drag and drop host will do
1625  // the OS dependent code to "lift off the dragged item".
1626  DCHECK(!IsDraggingForReparentInRootLevelGridView());
1627  drag_and_drop_host_->CreateDragIconProxy(screen_location,
1628                                           drag_view_->item()->icon(),
1629                                           drag_view_,
1630                                           delta,
1631                                           kDragAndDropProxyScale);
1632  SetViewHidden(drag_view_,
1633           true /* hide */,
1634           true /* no animation */);
1635}
1636
1637void AppsGridView::DispatchDragEventToDragAndDropHost(
1638    const gfx::Point& location_in_screen_coordinates) {
1639  if (!drag_view_ || !drag_and_drop_host_)
1640    return;
1641
1642  if (GetLocalBounds().Contains(last_drag_point_)) {
1643    // The event was issued inside the app menu and we should get all events.
1644    if (forward_events_to_drag_and_drop_host_) {
1645      // The DnD host was previously called and needs to be informed that the
1646      // session returns to the owner.
1647      forward_events_to_drag_and_drop_host_ = false;
1648      drag_and_drop_host_->EndDrag(true);
1649    }
1650  } else {
1651    if (IsFolderItem(drag_view_->item()))
1652      return;
1653
1654    // The event happened outside our app menu and we might need to dispatch.
1655    if (forward_events_to_drag_and_drop_host_) {
1656      // Dispatch since we have already started.
1657      if (!drag_and_drop_host_->Drag(location_in_screen_coordinates)) {
1658        // The host is not active any longer and we cancel the operation.
1659        forward_events_to_drag_and_drop_host_ = false;
1660        drag_and_drop_host_->EndDrag(true);
1661      }
1662    } else {
1663      if (drag_and_drop_host_->StartDrag(drag_view_->item()->id(),
1664                                         location_in_screen_coordinates)) {
1665        // From now on we forward the drag events.
1666        forward_events_to_drag_and_drop_host_ = true;
1667        // Any flip operations are stopped.
1668        StopPageFlipTimer();
1669      }
1670    }
1671  }
1672}
1673
1674void AppsGridView::MaybeStartPageFlipTimer(const gfx::Point& drag_point) {
1675  if (!IsPointWithinDragBuffer(drag_point))
1676    StopPageFlipTimer();
1677  int new_page_flip_target = -1;
1678
1679  // Drag zones are at the edges of the scroll axis.
1680  if (pagination_controller_->scroll_axis() ==
1681      PaginationController::SCROLL_AXIS_VERTICAL) {
1682    if (drag_point.y() < kPageFlipZoneSize)
1683      new_page_flip_target = pagination_model_.selected_page() - 1;
1684    else if (drag_point.y() > height() - kPageFlipZoneSize)
1685      new_page_flip_target = pagination_model_.selected_page() + 1;
1686  } else {
1687    if (page_switcher_view_->bounds().Contains(drag_point)) {
1688      gfx::Point page_switcher_point(drag_point);
1689      views::View::ConvertPointToTarget(
1690          this, page_switcher_view_, &page_switcher_point);
1691      new_page_flip_target =
1692          page_switcher_view_->GetPageForPoint(page_switcher_point);
1693    }
1694
1695    // TODO(xiyuan): Fix this for RTL.
1696    if (new_page_flip_target == -1 && drag_point.x() < kPageFlipZoneSize)
1697      new_page_flip_target = pagination_model_.selected_page() - 1;
1698
1699    if (new_page_flip_target == -1 &&
1700        drag_point.x() > width() - kPageFlipZoneSize) {
1701      new_page_flip_target = pagination_model_.selected_page() + 1;
1702    }
1703  }
1704
1705  if (new_page_flip_target == page_flip_target_)
1706    return;
1707
1708  StopPageFlipTimer();
1709  if (pagination_model_.is_valid_page(new_page_flip_target)) {
1710    page_flip_target_ = new_page_flip_target;
1711
1712    if (page_flip_target_ != pagination_model_.selected_page()) {
1713      page_flip_timer_.Start(FROM_HERE,
1714          base::TimeDelta::FromMilliseconds(page_flip_delay_in_ms_),
1715          this, &AppsGridView::OnPageFlipTimer);
1716    }
1717  }
1718}
1719
1720void AppsGridView::OnPageFlipTimer() {
1721  DCHECK(pagination_model_.is_valid_page(page_flip_target_));
1722  pagination_model_.SelectPage(page_flip_target_, true);
1723}
1724
1725void AppsGridView::MoveItemInModel(views::View* item_view,
1726                                   const Index& target) {
1727  int current_model_index = view_model_.GetIndexOfView(item_view);
1728  DCHECK_GE(current_model_index, 0);
1729
1730  int target_model_index = GetModelIndexFromIndex(target);
1731  if (target_model_index == current_model_index)
1732    return;
1733
1734  item_list_->RemoveObserver(this);
1735  item_list_->MoveItem(current_model_index, target_model_index);
1736  view_model_.Move(current_model_index, target_model_index);
1737  item_list_->AddObserver(this);
1738
1739  if (pagination_model_.selected_page() != target.page)
1740    pagination_model_.SelectPage(target.page, false);
1741}
1742
1743void AppsGridView::MoveItemToFolder(views::View* item_view,
1744                                    const Index& target) {
1745  const std::string& source_item_id =
1746      static_cast<AppListItemView*>(item_view)->item()->id();
1747  AppListItemView* target_view =
1748      static_cast<AppListItemView*>(GetViewAtSlotOnCurrentPage(target.slot));
1749  const std::string&  target_view_item_id = target_view->item()->id();
1750
1751  // Make change to data model.
1752  item_list_->RemoveObserver(this);
1753  std::string folder_item_id =
1754      model_->MergeItems(target_view_item_id, source_item_id);
1755  item_list_->AddObserver(this);
1756  if (folder_item_id.empty()) {
1757    LOG(ERROR) << "Unable to merge into item id: " << target_view_item_id;
1758    return;
1759  }
1760  if (folder_item_id != target_view_item_id) {
1761    // New folder was created, change the view model to replace the old target
1762    // view with the new folder item view.
1763    size_t folder_item_index;
1764    if (item_list_->FindItemIndex(folder_item_id, &folder_item_index)) {
1765      int target_view_index = view_model_.GetIndexOfView(target_view);
1766      gfx::Rect target_view_bounds = target_view->bounds();
1767      DeleteItemViewAtIndex(target_view_index);
1768      views::View* target_folder_view =
1769          CreateViewForItemAtIndex(folder_item_index);
1770      target_folder_view->SetBoundsRect(target_view_bounds);
1771      view_model_.Add(target_folder_view, target_view_index);
1772      AddChildView(target_folder_view);
1773    } else {
1774      LOG(ERROR) << "Folder no longer in item_list: " << folder_item_id;
1775    }
1776  }
1777
1778  // Fade out the drag_view_ and delete it when animation ends.
1779  int drag_view_index = view_model_.GetIndexOfView(drag_view_);
1780  view_model_.Remove(drag_view_index);
1781  bounds_animator_.AnimateViewTo(drag_view_, drag_view_->bounds());
1782  bounds_animator_.SetAnimationDelegate(
1783      drag_view_,
1784      scoped_ptr<gfx::AnimationDelegate>(
1785          new ItemRemoveAnimationDelegate(drag_view_)));
1786  UpdatePaging();
1787}
1788
1789void AppsGridView::ReparentItemForReorder(views::View* item_view,
1790                                          const Index& target) {
1791  item_list_->RemoveObserver(this);
1792  model_->RemoveObserver(this);
1793
1794  AppListItem* reparent_item = static_cast<AppListItemView*>(item_view)->item();
1795  DCHECK(reparent_item->IsInFolder());
1796  const std::string source_folder_id = reparent_item->folder_id();
1797  AppListFolderItem* source_folder =
1798      static_cast<AppListFolderItem*>(item_list_->FindItem(source_folder_id));
1799
1800  int target_model_index = GetModelIndexFromIndex(target);
1801
1802  // Remove the source folder view if there is only 1 item in it, since the
1803  // source folder will be deleted after its only child item removed from it.
1804  if (source_folder->ChildItemCount() == 1u) {
1805    const int deleted_folder_index =
1806        view_model_.GetIndexOfView(activated_folder_item_view());
1807    DeleteItemViewAtIndex(deleted_folder_index);
1808
1809    // Adjust |target_model_index| if it is beyond the deleted folder index.
1810    if (target_model_index > deleted_folder_index)
1811      --target_model_index;
1812  }
1813
1814  // Move the item from its parent folder to top level item list.
1815  // Must move to target_model_index, the location we expect the target item
1816  // to be, not the item location we want to insert before.
1817  int current_model_index = view_model_.GetIndexOfView(item_view);
1818  syncer::StringOrdinal target_position;
1819  if (target_model_index < static_cast<int>(item_list_->item_count()))
1820    target_position = item_list_->item_at(target_model_index)->position();
1821  model_->MoveItemToFolderAt(reparent_item, "", target_position);
1822  view_model_.Move(current_model_index, target_model_index);
1823
1824  RemoveLastItemFromReparentItemFolderIfNecessary(source_folder_id);
1825
1826  item_list_->AddObserver(this);
1827  model_->AddObserver(this);
1828  UpdatePaging();
1829}
1830
1831void AppsGridView::ReparentItemToAnotherFolder(views::View* item_view,
1832                                               const Index& target) {
1833  DCHECK(IsDraggingForReparentInRootLevelGridView());
1834
1835  AppListItemView* target_view =
1836      static_cast<AppListItemView*>(GetViewAtSlotOnCurrentPage(target.slot));
1837  if (!target_view)
1838    return;
1839
1840  // Make change to data model.
1841  item_list_->RemoveObserver(this);
1842
1843  AppListItem* reparent_item = static_cast<AppListItemView*>(item_view)->item();
1844  DCHECK(reparent_item->IsInFolder());
1845  const std::string source_folder_id = reparent_item->folder_id();
1846  AppListFolderItem* source_folder =
1847      static_cast<AppListFolderItem*>(item_list_->FindItem(source_folder_id));
1848
1849  // Remove the source folder view if there is only 1 item in it, since the
1850  // source folder will be deleted after its only child item merged into the
1851  // target item.
1852  if (source_folder->ChildItemCount() == 1u)
1853    DeleteItemViewAtIndex(
1854        view_model_.GetIndexOfView(activated_folder_item_view()));
1855
1856  AppListItem* target_item = target_view->item();
1857
1858  // Move item to the target folder.
1859  std::string target_id_after_merge =
1860      model_->MergeItems(target_item->id(), reparent_item->id());
1861  if (target_id_after_merge.empty()) {
1862    LOG(ERROR) << "Unable to reparent to item id: " << target_item->id();
1863    item_list_->AddObserver(this);
1864    return;
1865  }
1866
1867  if (target_id_after_merge != target_item->id()) {
1868    // New folder was created, change the view model to replace the old target
1869    // view with the new folder item view.
1870    const std::string& new_folder_id = reparent_item->folder_id();
1871    size_t new_folder_index;
1872    if (item_list_->FindItemIndex(new_folder_id, &new_folder_index)) {
1873      int target_view_index = view_model_.GetIndexOfView(target_view);
1874      DeleteItemViewAtIndex(target_view_index);
1875      views::View* new_folder_view =
1876          CreateViewForItemAtIndex(new_folder_index);
1877      view_model_.Add(new_folder_view, target_view_index);
1878      AddChildView(new_folder_view);
1879    } else {
1880      LOG(ERROR) << "Folder no longer in item_list: " << new_folder_id;
1881    }
1882  }
1883
1884  RemoveLastItemFromReparentItemFolderIfNecessary(source_folder_id);
1885
1886  item_list_->AddObserver(this);
1887
1888  // Fade out the drag_view_ and delete it when animation ends.
1889  int drag_view_index = view_model_.GetIndexOfView(drag_view_);
1890  view_model_.Remove(drag_view_index);
1891  bounds_animator_.AnimateViewTo(drag_view_, drag_view_->bounds());
1892  bounds_animator_.SetAnimationDelegate(
1893      drag_view_,
1894      scoped_ptr<gfx::AnimationDelegate>(
1895          new ItemRemoveAnimationDelegate(drag_view_)));
1896  UpdatePaging();
1897}
1898
1899// After moving the re-parenting item out of the folder, if there is only 1 item
1900// left, remove the last item out of the folder, delete the folder and insert it
1901// to the data model at the same position. Make the same change to view_model_
1902// accordingly.
1903void AppsGridView::RemoveLastItemFromReparentItemFolderIfNecessary(
1904    const std::string& source_folder_id) {
1905  AppListFolderItem* source_folder =
1906      static_cast<AppListFolderItem*>(item_list_->FindItem(source_folder_id));
1907  if (!source_folder || source_folder->ChildItemCount() != 1u)
1908    return;
1909
1910  // Delete view associated with the folder item to be removed.
1911  DeleteItemViewAtIndex(
1912      view_model_.GetIndexOfView(activated_folder_item_view()));
1913
1914  // Now make the data change to remove the folder item in model.
1915  AppListItem* last_item = source_folder->item_list()->item_at(0);
1916  model_->MoveItemToFolderAt(last_item, "", source_folder->position());
1917
1918  // Create a new item view for the last item in folder.
1919  size_t last_item_index;
1920  if (!item_list_->FindItemIndex(last_item->id(), &last_item_index) ||
1921      last_item_index > static_cast<size_t>(view_model_.view_size())) {
1922    NOTREACHED();
1923    return;
1924  }
1925  views::View* last_item_view = CreateViewForItemAtIndex(last_item_index);
1926  view_model_.Add(last_item_view, last_item_index);
1927  AddChildView(last_item_view);
1928}
1929
1930void AppsGridView::CancelFolderItemReparent(AppListItemView* drag_item_view) {
1931  // The icon of the dragged item must target to its final ideal bounds after
1932  // the animation completes.
1933  CalculateIdealBounds();
1934
1935  gfx::Rect target_icon_rect =
1936      GetTargetIconRectInFolder(drag_item_view, activated_folder_item_view_);
1937
1938  gfx::Rect drag_view_icon_to_grid =
1939      drag_item_view->ConvertRectToParent(drag_item_view->GetIconBounds());
1940  drag_view_icon_to_grid.ClampToCenteredSize(
1941      gfx::Size(kGridIconDimension, kGridIconDimension));
1942  TopIconAnimationView* icon_view = new TopIconAnimationView(
1943      drag_item_view->item()->icon(),
1944      target_icon_rect,
1945      false);    /* animate like closing folder */
1946  AddChildView(icon_view);
1947  icon_view->SetBoundsRect(drag_view_icon_to_grid);
1948  icon_view->TransformView();
1949}
1950
1951void AppsGridView::CancelContextMenusOnCurrentPage() {
1952  int start = pagination_model_.selected_page() * tiles_per_page();
1953  int end = std::min(view_model_.view_size(), start + tiles_per_page());
1954  for (int i = start; i < end; ++i) {
1955    AppListItemView* view =
1956        static_cast<AppListItemView*>(view_model_.view_at(i));
1957    view->CancelContextMenu();
1958  }
1959}
1960
1961void AppsGridView::DeleteItemViewAtIndex(int index) {
1962  views::View* item_view = view_model_.view_at(index);
1963  view_model_.Remove(index);
1964  if (item_view == drag_view_)
1965    drag_view_ = NULL;
1966  delete item_view;
1967}
1968
1969bool AppsGridView::IsPointWithinDragBuffer(const gfx::Point& point) const {
1970  gfx::Rect rect(GetLocalBounds());
1971  rect.Inset(-kDragBufferPx, -kDragBufferPx, -kDragBufferPx, -kDragBufferPx);
1972  return rect.Contains(point);
1973}
1974
1975void AppsGridView::ButtonPressed(views::Button* sender,
1976                                 const ui::Event& event) {
1977  if (dragging())
1978    return;
1979
1980  if (strcmp(sender->GetClassName(), AppListItemView::kViewClassName))
1981    return;
1982
1983  if (delegate_) {
1984    // Always set the previous activated_folder_item_view_ to be visible. This
1985    // prevents a case where the item would remain hidden due the
1986    // |activated_folder_item_view_| changing during the animation. We only
1987    // need to track |activated_folder_item_view_| in the root level grid view.
1988    if (!folder_delegate_) {
1989      if (activated_folder_item_view_)
1990        activated_folder_item_view_->SetVisible(true);
1991      AppListItemView* pressed_item_view =
1992          static_cast<AppListItemView*>(sender);
1993      if (IsFolderItem(pressed_item_view->item()))
1994        activated_folder_item_view_ = pressed_item_view;
1995      else
1996        activated_folder_item_view_ = NULL;
1997    }
1998    delegate_->ActivateApp(static_cast<AppListItemView*>(sender)->item(),
1999                           event.flags());
2000  }
2001}
2002
2003void AppsGridView::OnListItemAdded(size_t index, AppListItem* item) {
2004  EndDrag(true);
2005
2006  views::View* view = CreateViewForItemAtIndex(index);
2007  view_model_.Add(view, index);
2008  AddChildView(view);
2009
2010  UpdatePaging();
2011  UpdatePulsingBlockViews();
2012  Layout();
2013  SchedulePaint();
2014}
2015
2016void AppsGridView::OnListItemRemoved(size_t index, AppListItem* item) {
2017  EndDrag(true);
2018
2019  DeleteItemViewAtIndex(index);
2020
2021  UpdatePaging();
2022  UpdatePulsingBlockViews();
2023  Layout();
2024  SchedulePaint();
2025}
2026
2027void AppsGridView::OnListItemMoved(size_t from_index,
2028                                   size_t to_index,
2029                                   AppListItem* item) {
2030  EndDrag(true);
2031  view_model_.Move(from_index, to_index);
2032
2033  UpdatePaging();
2034  AnimateToIdealBounds();
2035}
2036
2037void AppsGridView::TotalPagesChanged() {
2038}
2039
2040void AppsGridView::SelectedPageChanged(int old_selected, int new_selected) {
2041  if (dragging()) {
2042    CalculateDropTarget();
2043    Layout();
2044    MaybeStartPageFlipTimer(last_drag_point_);
2045  } else {
2046    ClearSelectedView(selected_view_);
2047    Layout();
2048  }
2049}
2050
2051void AppsGridView::TransitionStarted() {
2052  CancelContextMenusOnCurrentPage();
2053}
2054
2055void AppsGridView::TransitionChanged() {
2056  // Update layout for valid page transition only since over-scroll no longer
2057  // animates app icons.
2058  const PaginationModel::Transition& transition =
2059      pagination_model_.transition();
2060  if (pagination_model_.is_valid_page(transition.target_page))
2061    Layout();
2062}
2063
2064void AppsGridView::OnAppListModelStatusChanged() {
2065  UpdatePulsingBlockViews();
2066  Layout();
2067  SchedulePaint();
2068}
2069
2070void AppsGridView::SetViewHidden(views::View* view, bool hide, bool immediate) {
2071  ui::ScopedLayerAnimationSettings animator(view->layer()->GetAnimator());
2072  animator.SetPreemptionStrategy(
2073      immediate ? ui::LayerAnimator::IMMEDIATELY_SET_NEW_TARGET :
2074                  ui::LayerAnimator::BLEND_WITH_CURRENT_ANIMATION);
2075  view->layer()->SetOpacity(hide ? 0 : 1);
2076}
2077
2078void AppsGridView::OnImplicitAnimationsCompleted() {
2079  if (layer()->opacity() == 0.0f)
2080    SetVisible(false);
2081}
2082
2083bool AppsGridView::EnableFolderDragDropUI() {
2084  // Enable drag and drop folder UI only if it is at the app list root level
2085  // and the switch is on.
2086  return model_->folders_enabled() && !folder_delegate_;
2087}
2088
2089bool AppsGridView::CanDropIntoTarget(const Index& drop_target) const {
2090  views::View* target_view = GetViewAtSlotOnCurrentPage(drop_target.slot);
2091
2092  if (!target_view)
2093    return false;
2094
2095  AppListItem* target_item =
2096      static_cast<AppListItemView*>(target_view)->item();
2097  // Items can be dropped into non-folders (which have no children) or folders
2098  // that have fewer than the max allowed items.
2099  // OEM folder does not allow to drag/drop other items in it.
2100  return target_item->ChildItemCount() < kMaxFolderItems &&
2101         !IsOEMFolderItem(target_item);
2102}
2103
2104AppsGridView::Index AppsGridView::GetNearestTileIndexForPoint(
2105    const gfx::Point& point) const {
2106  gfx::Rect bounds = GetContentsBounds();
2107  gfx::Size total_tile_size = GetTotalTileSize();
2108  int col = ClampToRange(
2109      (point.x() - bounds.x()) / total_tile_size.width(), 0, cols_ - 1);
2110  int row = ClampToRange((point.y() - bounds.y()) / total_tile_size.height(),
2111                         0,
2112                         rows_per_page_ - 1);
2113  return Index(pagination_model_.selected_page(), row * cols_ + col);
2114}
2115
2116gfx::Size AppsGridView::GetTileGridSize() const {
2117  gfx::Rect bounds = GetExpectedTileBounds(0, 0);
2118  bounds.Union(GetExpectedTileBounds(rows_per_page_ - 1, cols_ - 1));
2119  if (switches::IsExperimentalAppListEnabled())
2120    bounds.Inset(-kExperimentalTileLeftRightPadding,
2121                 -kExperimentalTileTopBottomPadding);
2122  return bounds.size();
2123}
2124
2125gfx::Rect AppsGridView::GetExpectedTileBounds(int slot) const {
2126  return GetExpectedTileBounds(slot / cols_, slot % cols_);
2127}
2128
2129gfx::Rect AppsGridView::GetExpectedTileBounds(int row, int col) const {
2130  gfx::Rect bounds(GetContentsBounds());
2131  gfx::Size total_tile_size = GetTotalTileSize();
2132  gfx::Rect tile_bounds(gfx::Point(bounds.x() + col * total_tile_size.width(),
2133                                   bounds.y() + row * total_tile_size.height()),
2134                        total_tile_size);
2135  tile_bounds.ClampToCenteredSize(GetTileViewSize());
2136  return tile_bounds;
2137}
2138
2139views::View* AppsGridView::GetViewAtSlotOnCurrentPage(int slot) const {
2140  if (slot < 0)
2141    return NULL;
2142
2143  // Calculate the original bound of the tile at |index|.
2144  int row = slot / cols_;
2145  int col = slot % cols_;
2146  gfx::Rect tile_rect = GetExpectedTileBounds(row, col);
2147
2148  for (int i = 0; i < view_model_.view_size(); ++i) {
2149    views::View* view = view_model_.view_at(i);
2150    if (view->bounds() == tile_rect && view != drag_view_)
2151      return view;
2152  }
2153  return NULL;
2154}
2155
2156void AppsGridView::SetAsFolderDroppingTarget(const Index& target_index,
2157                                             bool is_target_folder) {
2158  AppListItemView* target_view =
2159      static_cast<AppListItemView*>(
2160          GetViewAtSlotOnCurrentPage(target_index.slot));
2161  if (target_view)
2162    target_view->SetAsAttemptedFolderTarget(is_target_folder);
2163}
2164
2165}  // namespace app_list
2166