//The MIT License (MIT)
//
//Copyright (c) 2014 RafaƂ Augustyniak
//
//Permission is hereby granted, free of charge, to any person obtaining a copy of
//this software and associated documentation files (the "Software"), to deal in
//the Software without restriction, including without limitation the rights to
//use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
//the Software, and to permit persons to whom the Software is furnished to do so,
//subject to the following conditions:
//
//THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
//IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
//FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
//COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
//IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
//CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//

#import "RATreeView+Private.h"
#import "RATreeView+Enums.h"
#import "RATreeView+RATreeNodeCollectionControllerDataSource.h"
#import "RATreeView_ClassExtension.h"

#import "RABatchChanges.h"

#import "RATreeNode.h"
#import "RATreeNodeController.h"
#import "RATreeNodeCollectionController.h"



@implementation RATreeView (Private)

- (void)setupTreeStructure
{
  self.treeNodeCollectionController = [[RATreeNodeCollectionController alloc] init];
  self.treeNodeCollectionController.dataSource = self;
  self.batchChanges = [[RABatchChanges alloc] init];
}

- (NSArray *)childrenForItem:(id)item
{
  NSParameterAssert(item);
  
  NSMutableArray *children = [NSMutableArray array];
  NSInteger numberOfChildren = [self.dataSource treeView:self numberOfChildrenOfItem:item];
  
  for (int i = 0; i < numberOfChildren; i++) {
    [children addObject:[self.dataSource treeView:self child:i ofItem:item]];
  }
  
  return [NSArray arrayWithArray:children];
}

- (RATreeNode *)treeNodeForIndexPath:(NSIndexPath *)indexPath
{
  NSParameterAssert(indexPath.section == 0);
  return [self.treeNodeCollectionController treeNodeForIndex:indexPath.row];
}

- (NSIndexPath *)indexPathForItem:(id)item
{
  return [NSIndexPath indexPathForRow:[self.treeNodeCollectionController indexForItem:item] inSection:0];
}


#pragma mark Collapsing and Expanding Rows

- (void)collapseCellForTreeNode:(RATreeNode *)treeNode
{
  [self collapseCellForTreeNode:treeNode collapseChildren:self.collapsesChildRowsWhenRowCollapses withRowAnimation:self.rowsCollapsingAnimation];
}

- (void)collapseCellForTreeNode:(RATreeNode *)treeNode collapseChildren:(BOOL)collapseChildren withRowAnimation:(RATreeViewRowAnimation)rowAnimation
{
  [self.tableView beginUpdates];
  [self.batchChanges beginUpdates];
  
  NSInteger index = [self.treeNodeCollectionController lastVisibleDescendantIndexForItem:treeNode.item];
  
  __weak __typeof(self) weakSelf = self;
  [self.batchChanges collapseItemWithBlock:^{
    UITableViewRowAnimation tableViewRowAnimation = [RATreeView tableViewRowAnimationForTreeViewRowAnimation:rowAnimation];
    [weakSelf.treeNodeCollectionController collapseRowForItem:treeNode.item collapseChildren:collapseChildren updates:^(NSIndexSet *deletions) {
      [weakSelf.tableView deleteRowsAtIndexPaths:IndexesToIndexPaths(deletions) withRowAnimation:tableViewRowAnimation];
    }];
  } lastIndex:index];
  
  [self.batchChanges endUpdates];
  [self.tableView endUpdates];
}

- (void)expandCellForTreeNode:(RATreeNode *)treeNode
{
  [self expandCellForTreeNode:treeNode expandChildren:self.expandsChildRowsWhenRowExpands withRowAnimation:self.rowsExpandingAnimation];
}

- (void)expandCellForTreeNode:(RATreeNode *)treeNode expandChildren:(BOOL)expandChildren withRowAnimation:(RATreeViewRowAnimation)rowAnimation
{
  [self.tableView beginUpdates];
  [self.batchChanges beginUpdates];
  
  NSInteger index = [self.treeNodeCollectionController indexForItem:treeNode.item];
  __weak __typeof(self) weakSelf = self;
  [self.batchChanges expandItemWithBlock:^{
    UITableViewRowAnimation tableViewRowAnimation = [RATreeView tableViewRowAnimationForTreeViewRowAnimation:rowAnimation];
    [weakSelf.treeNodeCollectionController expandRowForItem:treeNode.item expandChildren:expandChildren updates:^(NSIndexSet *insertions) {
      [weakSelf.tableView insertRowsAtIndexPaths:IndexesToIndexPaths(insertions) withRowAnimation:tableViewRowAnimation];
    }];
  } atIndex:index];
  
  
  [self.batchChanges endUpdates];
  [self.tableView endUpdates];
}

- (void)insertItemAtIndex:(NSInteger)index inParent:(id)parent withAnimation:(RATreeViewRowAnimation)animation
{
  NSInteger idx = [self.treeNodeCollectionController indexForItem:parent];
  if (idx == NSNotFound) {
    return;
  }
  idx += index + 1;
  
  __weak __typeof(self) weakSelf = self;
  [self.batchChanges insertItemWithBlock:^{
    [weakSelf.treeNodeCollectionController insertItemsAtIndexes:[NSIndexSet indexSetWithIndex:index] inParent:parent];
    
    NSIndexPath *indexPath = [NSIndexPath indexPathForRow:idx inSection:0];
    UITableViewRowAnimation tableViewRowAnimation = [RATreeView tableViewRowAnimationForTreeViewRowAnimation:animation];
    [weakSelf.tableView insertRowsAtIndexPaths:@[indexPath] withRowAnimation:tableViewRowAnimation];
    
  } atIndex:idx];
}

#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wobjc-protocol-method-implementation"
- (void)moveItemAtIndex:(NSInteger)index inParent:(id)parent toIndex:(NSInteger)newIndex inParent:(id)newParent
#pragma clang diagnostic pop
{
  NSInteger idx = [self.treeNodeCollectionController indexForItem:parent];
  if (idx == NSNotFound) {
    return;
  }
  
  idx += index + 1;
  __weak __typeof(self) weakSelf = self;
  [self.batchChanges insertItemWithBlock:^{
    [weakSelf.treeNodeCollectionController moveItemAtIndex:index inParent:parent toIndex:newIndex inParent:newParent updates:^(NSIndexSet *deletions, NSIndexSet *additions) {
      NSArray *deletionsArray = IndexesToIndexPaths(deletions);
      NSArray *additionsArray = IndexesToIndexPaths(additions);
    
      NSInteger i = 0;
      for (NSIndexPath *deletedIndexPath in deletionsArray) {
        [weakSelf.tableView moveRowAtIndexPath:deletedIndexPath toIndexPath:additionsArray[i]];
        i++;
      }
    }];
  } atIndex:idx];
}

- (void)removeItemAtIndex:(NSInteger)index inParent:(id)parent withAnimation:(RATreeViewRowAnimation)animation
{
  id child = [self.treeNodeCollectionController childInParent:parent atIndex:index];
  NSInteger idx = [self.treeNodeCollectionController lastVisibleDescendantIndexForItem:child];
  if (idx == NSNotFound) {
    return;
  }
  
  __weak __typeof(self) weakSelf = self;
  [self.batchChanges insertItemWithBlock:^{
    [weakSelf.treeNodeCollectionController removeItemsAtIndexes:[NSIndexSet indexSetWithIndex:index] inParent:parent updates:^(NSIndexSet *removedIndexes) {
      UITableViewRowAnimation tableViewRowAnimation = [RATreeView tableViewRowAnimationForTreeViewRowAnimation:animation];
      [weakSelf.tableView deleteRowsAtIndexPaths:IndexesToIndexPaths(removedIndexes) withRowAnimation:tableViewRowAnimation];
    }];
  } atIndex:idx];
}

#pragma mark -

static NSArray * IndexesToIndexPaths(NSIndexSet *indexes)
{
  NSMutableArray *indexPaths = [NSMutableArray array];
  [indexes enumerateIndexesUsingBlock:^(NSUInteger idx, BOOL *stop) {
    [indexPaths addObject:[NSIndexPath indexPathForRow:idx inSection:0]];
  }];
  return [indexPaths copy];
}

@end