Home | History | Annotate | Download | only in views
      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 
     52 namespace app_list {
     53 
     54 namespace {
     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.
     60 const int kDragBufferPx = 20;
     61 
     62 // Padding space in pixels for fixed layout.
     63 const int kLeftRightPadding = 23;
     64 const int kTopPadding = 1;
     65 
     66 // Padding space in pixels between pages.
     67 const int kPagePadding = 40;
     68 
     69 // Preferred tile size when showing in fixed layout.
     70 const int kPreferredTileWidth = 88;
     71 const int kPreferredTileHeight = 98;
     72 
     73 const int kExperimentalPreferredTileWidth = 90;
     74 const int kExperimentalPrefferedTileHeight = 90;
     75 
     76 // Padding on each side of a tile.
     77 const int kExperimentalTileLeftRightPadding = 15;
     78 const int kExperimentalTileTopBottomPadding = 11;
     79 
     80 // Width in pixels of the area on the sides that triggers a page flip.
     81 const int kPageFlipZoneSize = 40;
     82 
     83 // Delay in milliseconds to do the page flip.
     84 const int kPageFlipDelayInMs = 1000;
     85 
     86 // How many pages on either side of the selected one we prerender.
     87 const int kPrerenderPages = 1;
     88 
     89 // The drag and drop proxy should get scaled by this factor.
     90 const float kDragAndDropProxyScale = 1.5f;
     91 
     92 // Delays in milliseconds to show folder dropping preview circle.
     93 const int kFolderDroppingDelay = 150;
     94 
     95 // Delays in milliseconds to show re-order preview.
     96 const int kReorderDelay = 120;
     97 
     98 // Delays in milliseconds to show folder item reparent UI.
     99 const int kFolderItemReparentDelay = 50;
    100 
    101 // Radius of the circle, in which if entered, show folder dropping preview
    102 // UI.
    103 const int kFolderDroppingCircleRadius = 39;
    104 
    105 // Returns the size of a tile view excluding its padding.
    106 gfx::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.
    114 gfx::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.
    127 class 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.
    174 class 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.
    198 class 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.
    216 bool IsFolderItem(AppListItem* item) {
    217   return (item->GetItemType() == AppListFolderItem::kItemType);
    218 }
    219 
    220 bool IsOEMFolderItem(AppListItem* item) {
    221   return IsFolderItem(item) &&
    222          (static_cast<AppListFolderItem*>(item))->folder_type() ==
    223              AppListFolderItem::FOLDER_TYPE_OEM;
    224 }
    225 
    226 int 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.
    242 class 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 
    357 AppsGridView::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 
    403 AppsGridView::~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 
    423 void 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 
    434 void 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 
    448 void 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 
    459 void 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 
    468 void 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 
    477 void AppsGridView::ClearSelectedView(views::View* view) {
    478   if (view && IsSelectedView(view)) {
    479     selected_view_->SchedulePaint();
    480     selected_view_ = NULL;
    481   }
    482 }
    483 
    484 void AppsGridView::ClearAnySelectedView() {
    485   if (selected_view_) {
    486     selected_view_->SchedulePaint();
    487     selected_view_ = NULL;
    488   }
    489 }
    490 
    491 bool AppsGridView::IsSelectedView(const views::View* view) const {
    492   return selected_view_ == view;
    493 }
    494 
    495 void 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 
    504 void 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 
    520 void 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 
    546 bool 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 
    567 void 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)
    577 void 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 
    593 bool 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 
    614 void 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 
    680 void 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 
    759 void AppsGridView::StopPageFlipTimer() {
    760   page_flip_timer_.Stop();
    761   page_flip_target_ = -1;
    762 }
    763 
    764 AppListItemView* 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 
    769 void 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 
    776 void 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 
    794 void 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 
    834 void 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 
    845 bool AppsGridView::IsDraggedView(const views::View* view) const {
    846   return drag_view_ == view;
    847 }
    848 
    849 void 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 
    871 void AppsGridView::SetDragViewVisible(bool visible) {
    872   DCHECK(drag_view_);
    873   SetViewHidden(drag_view_, !visible, true);
    874 }
    875 
    876 void AppsGridView::SetDragAndDropHostOfCurrentAppList(
    877     ApplicationDragAndDropHost* drag_and_drop_host) {
    878   drag_and_drop_host_ = drag_and_drop_host;
    879 }
    880 
    881 void 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 
    893 bool AppsGridView::IsAnimatingView(views::View* view) {
    894   return bounds_animator_.IsAnimating(view);
    895 }
    896 
    897 gfx::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 
    907 bool 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 
    915 bool AppsGridView::CanDrop(const OSExchangeData& data) {
    916   return true;
    917 }
    918 
    919 int AppsGridView::OnDragUpdated(const ui::DropTargetEvent& event) {
    920   return ui::DragDropTypes::DRAG_MOVE;
    921 }
    922 
    923 void 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 
    945 bool 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 
    981 bool 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 
    989 bool AppsGridView::OnMouseWheel(const ui::MouseWheelEvent& event) {
    990   return pagination_controller_->OnScroll(
    991       gfx::Vector2d(event.x_offset(), event.y_offset()));
    992 }
    993 
    994 void 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 
   1012 void AppsGridView::OnGestureEvent(ui::GestureEvent* event) {
   1013   if (pagination_controller_->OnGestureEvent(*event, GetContentsBounds()))
   1014     event->SetHandled();
   1015 }
   1016 
   1017 void 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 
   1028 void 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 
   1044 void 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 
   1052 void 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 
   1075 views::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 
   1086 AppsGridView::Index AppsGridView::GetIndexFromModelIndex(
   1087     int model_index) const {
   1088   return Index(model_index / tiles_per_page(), model_index % tiles_per_page());
   1089 }
   1090 
   1091 int AppsGridView::GetModelIndexFromIndex(const Index& index) const {
   1092   return index.page * tiles_per_page() + index.slot;
   1093 }
   1094 
   1095 void 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 
   1113 bool 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 
   1119 AppsGridView::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 
   1128 views::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 
   1136 AppsGridView::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 
   1142 void 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 
   1184 void 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 
   1267 void 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 
   1304 void 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 
   1344 void 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 
   1370 void 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 
   1398 bool 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 
   1416 void 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 
   1449 void AppsGridView::OnReorderTimer() {
   1450   if (drop_attempt_ == DROP_FOR_REORDER) {
   1451     reorder_placeholder_ = reorder_drop_target_;
   1452     AnimateToIdealBounds();
   1453   }
   1454 }
   1455 
   1456 void 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 
   1470 void AppsGridView::OnFolderDroppingTimer() {
   1471   if (drop_attempt_ == DROP_FOR_FOLDER)
   1472     SetAsFolderDroppingTarget(folder_drop_target_, true);
   1473 }
   1474 
   1475 void 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 
   1511 bool AppsGridView::IsDraggingForReparentInRootLevelGridView() const {
   1512   return (!folder_delegate_ && dragging_for_reparent_item_);
   1513 }
   1514 
   1515 bool AppsGridView::IsDraggingForReparentInHiddenGridView() const {
   1516   return (folder_delegate_ && dragging_for_reparent_item_);
   1517 }
   1518 
   1519 gfx::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 
   1532 bool AppsGridView::IsUnderOEMFolder() {
   1533   if (!folder_delegate_)
   1534     return false;
   1535 
   1536   return folder_delegate_->IsOEMFolder();
   1537 }
   1538 
   1539 void AppsGridView::DispatchDragEventForReparent(Pointer pointer,
   1540                                                 const gfx::Point& drag_point) {
   1541   folder_delegate_->DispatchDragEventForReparent(pointer, drag_point);
   1542 }
   1543 
   1544 void 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 
   1587 void 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 
   1602 void AppsGridView::OnFolderItemRemoved() {
   1603   DCHECK(folder_delegate_);
   1604   item_list_ = NULL;
   1605 }
   1606 
   1607 void 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 
   1637 void 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 
   1674 void 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 
   1720 void AppsGridView::OnPageFlipTimer() {
   1721   DCHECK(pagination_model_.is_valid_page(page_flip_target_));
   1722   pagination_model_.SelectPage(page_flip_target_, true);
   1723 }
   1724 
   1725 void 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 
   1743 void 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 
   1789 void 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 
   1831 void 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.
   1903 void 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 
   1930 void 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 
   1951 void 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 
   1961 void 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 
   1969 bool 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 
   1975 void 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 
   2003 void 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 
   2016 void 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 
   2027 void 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 
   2037 void AppsGridView::TotalPagesChanged() {
   2038 }
   2039 
   2040 void 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 
   2051 void AppsGridView::TransitionStarted() {
   2052   CancelContextMenusOnCurrentPage();
   2053 }
   2054 
   2055 void 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 
   2064 void AppsGridView::OnAppListModelStatusChanged() {
   2065   UpdatePulsingBlockViews();
   2066   Layout();
   2067   SchedulePaint();
   2068 }
   2069 
   2070 void 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 
   2078 void AppsGridView::OnImplicitAnimationsCompleted() {
   2079   if (layer()->opacity() == 0.0f)
   2080     SetVisible(false);
   2081 }
   2082 
   2083 bool 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 
   2089 bool 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 
   2104 AppsGridView::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 
   2116 gfx::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 
   2125 gfx::Rect AppsGridView::GetExpectedTileBounds(int slot) const {
   2126   return GetExpectedTileBounds(slot / cols_, slot % cols_);
   2127 }
   2128 
   2129 gfx::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 
   2139 views::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 
   2156 void 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