// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#import <Cocoa/Cocoa.h>

#include "base/compiler_specific.h"
#include "base/strings/utf_string_conversions.h"
#include "chrome/browser/task_manager/resource_provider.h"
#import "chrome/browser/ui/cocoa/cocoa_test_helper.h"
#import "chrome/browser/ui/cocoa/task_manager_mac.h"
#include "chrome/grit/generated_resources.h"
#include "content/public/test/test_browser_thread_bundle.h"
#include "testing/gtest/include/gtest/gtest.h"
#import "testing/gtest_mac.h"
#include "testing/platform_test.h"
#include "ui/gfx/image/image_skia.h"

@interface TaskManagerWindowController(UnitTest)

- (void)toggleColumn:(NSMenuItem*)sender;

@end

namespace {

class TestResource : public task_manager::Resource {
 public:
  TestResource(const base::string16& title, pid_t pid)
      : title_(title), pid_(pid) {}
  base::string16 GetTitle() const override { return title_; }
  base::string16 GetProfileName() const override { return base::string16(); }
  gfx::ImageSkia GetIcon() const override { return gfx::ImageSkia(); }
  base::ProcessHandle GetProcess() const override { return pid_; }
  int GetUniqueChildProcessId() const override {
    // In reality the unique child process ID is not the actual process ID,
    // but for testing purposes it shouldn't make difference.
    return static_cast<int>(base::GetCurrentProcId());
  }
  Type GetType() const override { return RENDERER; }
  bool SupportNetworkUsage() const override { return false; }
  void SetSupportNetworkUsage() override { NOTREACHED(); }
  void Refresh() override {}
  base::string16 title_;
  base::string16 profile_name_;
  pid_t pid_;
};

}  // namespace

class TaskManagerWindowControllerTest : public CocoaTest {
  content::TestBrowserThreadBundle thread_bundle_;
};

// Test creation, to ensure nothing leaks or crashes.
TEST_F(TaskManagerWindowControllerTest, Init) {
  TaskManager task_manager;
  TaskManagerMac* bridge(new TaskManagerMac(&task_manager));
  TaskManagerWindowController* controller = bridge->cocoa_controller();

  // Releases the controller, which in turn deletes |bridge|.
  [controller close];
}

TEST_F(TaskManagerWindowControllerTest, Sort) {
  TaskManager task_manager;

  TestResource resource1(base::UTF8ToUTF16("zzz"), 1);
  TestResource resource2(base::UTF8ToUTF16("zzb"), 2);
  TestResource resource3(base::UTF8ToUTF16("zza"), 2);

  task_manager.AddResource(&resource1);
  task_manager.AddResource(&resource2);
  task_manager.AddResource(&resource3);  // Will be in the same group as 2.

  TaskManagerMac* bridge(new TaskManagerMac(&task_manager));
  TaskManagerWindowController* controller = bridge->cocoa_controller();
  NSTableView* table = [controller tableView];
  ASSERT_EQ(3, [controller numberOfRowsInTableView:table]);

  // Test that table is sorted on title.
  NSTableColumn* title_column = [table tableColumnWithIdentifier:
      [NSString stringWithFormat:@"%d", IDS_TASK_MANAGER_TASK_COLUMN]];
  NSCell* cell;
  cell = [controller tableView:table dataCellForTableColumn:title_column row:0];
  EXPECT_NSEQ(@"zzb", [cell title]);
  cell = [controller tableView:table dataCellForTableColumn:title_column row:1];
  EXPECT_NSEQ(@"zza", [cell title]);
  cell = [controller tableView:table dataCellForTableColumn:title_column row:2];
  EXPECT_NSEQ(@"zzz", [cell title]);

  // Releases the controller, which in turn deletes |bridge|.
  [controller close];

  task_manager.RemoveResource(&resource1);
  task_manager.RemoveResource(&resource2);
  task_manager.RemoveResource(&resource3);
}

TEST_F(TaskManagerWindowControllerTest, SelectionAdaptsToSorting) {
  TaskManager task_manager;

  TestResource resource1(base::UTF8ToUTF16("yyy"), 1);
  TestResource resource2(base::UTF8ToUTF16("aaa"), 2);

  task_manager.AddResource(&resource1);
  task_manager.AddResource(&resource2);

  TaskManagerMac* bridge(new TaskManagerMac(&task_manager));
  TaskManagerWindowController* controller = bridge->cocoa_controller();
  NSTableView* table = [controller tableView];
  ASSERT_EQ(2, [controller numberOfRowsInTableView:table]);

  // Select row 0 in the table (corresponds to row 1 in the model).
  [table selectRowIndexes:[NSIndexSet indexSetWithIndex:0]
     byExtendingSelection:NO];

  // Change the name of resource2 so that it becomes row 1 in the table.
  resource2.title_ = base::UTF8ToUTF16("zzz");
  bridge->task_manager()->model()->Refresh();
  bridge->OnItemsChanged(1, 1);

  // Check that the selection has moved to row 1.
  NSIndexSet* selection = [table selectedRowIndexes];
  ASSERT_EQ(1u, [selection count]);
  EXPECT_EQ(1u, [selection firstIndex]);

  // Releases the controller, which in turn deletes |bridge|.
  [controller close];

  task_manager.RemoveResource(&resource1);
  task_manager.RemoveResource(&resource2);
}

TEST_F(TaskManagerWindowControllerTest, EnsureNewPrimarySortColumn) {
  TaskManager task_manager;

  // Add a couple rows of data.
  TestResource resource1(base::UTF8ToUTF16("yyy"), 1);
  TestResource resource2(base::UTF8ToUTF16("aaa"), 2);

  task_manager.AddResource(&resource1);
  task_manager.AddResource(&resource2);

  TaskManagerMac* bridge(new TaskManagerMac(&task_manager));
  TaskManagerWindowController* controller = bridge->cocoa_controller();
  NSTableView* table = [controller tableView];
  ASSERT_EQ(2, [controller numberOfRowsInTableView:table]);

  // Locate the current first visible column.
  NSTableColumn* firstVisibleColumn = nil;
  for (NSTableColumn* nextColumn in [table tableColumns]) {
    if (![nextColumn isHidden]) {
      firstVisibleColumn = nextColumn;
      break;
    }
  }
  ASSERT_TRUE(firstVisibleColumn != nil);

  // Make the first visible column the primary sort column.
  NSSortDescriptor* sortDescriptor =
      [firstVisibleColumn sortDescriptorPrototype];
  ASSERT_TRUE(sortDescriptor != nil);
  [table setSortDescriptors:[NSArray arrayWithObject:sortDescriptor]];

  // Toggle the first visible column so that it's no longer visible, and make
  // sure a different column is now the primary sort column.
  NSMenuItem* menuItem = [[[NSMenuItem alloc]
                             initWithTitle:@"Temp"
                                    action:@selector(toggleColumn:)
                             keyEquivalent:@""] autorelease];
  [menuItem setRepresentedObject:firstVisibleColumn];
  [menuItem setState:NSOnState];
  [controller toggleColumn:menuItem];

  NSTableColumn* newFirstVisibleColumn = nil;
  for (NSTableColumn* nextColumn in [table tableColumns]) {
    if (![nextColumn isHidden]) {
      newFirstVisibleColumn = nextColumn;
      break;
    }
  }
  ASSERT_TRUE(newFirstVisibleColumn != nil);
  ASSERT_TRUE(newFirstVisibleColumn != firstVisibleColumn);
  NSSortDescriptor* newFirstSortDescriptor =
      [[table sortDescriptors] objectAtIndex:0];
  EXPECT_TRUE([newFirstSortDescriptor isEqual:
      [newFirstVisibleColumn sortDescriptorPrototype]]);

  // Release the controller, which in turn deletes |bridge|.
  [controller close];

  task_manager.RemoveResource(&resource1);
  task_manager.RemoveResource(&resource2);
}

TEST_F(TaskManagerWindowControllerTest, EnsureOneColumnVisible) {
  TaskManager task_manager;

  // Add a couple rows of data.
  TestResource resource1(base::UTF8ToUTF16("yyy"), 1);
  TestResource resource2(base::UTF8ToUTF16("aaa"), 2);

  task_manager.AddResource(&resource1);
  task_manager.AddResource(&resource2);

  TaskManagerMac* bridge(new TaskManagerMac(&task_manager));
  TaskManagerWindowController* controller = bridge->cocoa_controller();
  NSTableView* table = [controller tableView];
  ASSERT_EQ(2, [controller numberOfRowsInTableView:table]);

  // Toggle each visible column so that it's not visible.
  NSMenuItem* menuItem = [[[NSMenuItem alloc]
                             initWithTitle:@"Temp"
                                    action:@selector(toggleColumn:)
                             keyEquivalent:@""] autorelease];
  for (NSTableColumn* nextColumn in [table tableColumns]) {
    if (![nextColumn isHidden]) {
      [menuItem setState:NSOnState];
      [menuItem setRepresentedObject:nextColumn];
      [controller toggleColumn:menuItem];
    }
  }

  // Locate the one column that should still be visible.
  NSTableColumn* firstVisibleColumn = nil;
  for (NSTableColumn* nextColumn in [table tableColumns]) {
    if (![nextColumn isHidden]) {
      firstVisibleColumn = nextColumn;
      break;
    }
  }
  EXPECT_TRUE(firstVisibleColumn != nil);

  // Release the controller, which in turn deletes |bridge|.
  [controller close];

  task_manager.RemoveResource(&resource1);
  task_manager.RemoveResource(&resource2);
}