+class MockMediaGalleriesDialog
+ : public MediaGalleriesDialog {
+ public:
+ typedef base::Callback<void(int update_count)> DialogDestroyedCallback;
+
+ explicit MockMediaGalleriesDialog(const DialogDestroyedCallback& callback)
+ : update_count_(0),
+ dialog_destroyed_callback_(callback) {
+ }
+
+ virtual ~MockMediaGalleriesDialog() {
+ dialog_destroyed_callback_.Run(update_count_);
+ }
+
+ // MockMediaGalleriesDialog implementation.
+ virtual void UpdateGalleries() OVERRIDE {
+ update_count_++;
+ }
+
+ // Number of times UpdateResults has been called.
+ int update_count() {
+ return update_count_;
+ }
+
+ private:
+ int update_count_;
+
+ DialogDestroyedCallback dialog_destroyed_callback_;
+
+ DISALLOW_COPY_AND_ASSIGN(MockMediaGalleriesDialog);
+};
+
+} // namespace
+
+class MediaGalleriesDialogControllerTest : public ::testing::Test {
+ public:
+ MediaGalleriesDialogControllerTest()
+ : dialog_(NULL),
+ dialog_update_count_at_destruction_(0),
+ controller_(NULL),
+ profile_(new TestingProfile()),
+ weak_factory_(this) {
+ }
+
+ virtual ~MediaGalleriesDialogControllerTest() {
+ EXPECT_FALSE(controller_);
+ EXPECT_FALSE(dialog_);
+ }
+
+ virtual void SetUp() OVERRIDE {
+ ASSERT_TRUE(TestStorageMonitor::CreateAndInstall());
+
+ extensions::TestExtensionSystem* extension_system(
+ static_cast<extensions::TestExtensionSystem*>(
+ extensions::ExtensionSystem::Get(profile_.get())));
+ extension_system->CreateExtensionService(
+ CommandLine::ForCurrentProcess(), base::FilePath(), false);
+
+ gallery_prefs_.reset(new MediaGalleriesPreferences(profile_.get()));
+ base::RunLoop loop;
+ gallery_prefs_->EnsureInitialized(loop.QuitClosure());
+ loop.Run();
+
+ std::vector<std::string> read_permissions;
+ read_permissions.push_back(
+ extensions::MediaGalleriesPermission::kReadPermission);
+ extension_ = AddMediaGalleriesApp("read", read_permissions, profile_.get());
+ }
+
+ virtual void TearDown() OVERRIDE {
+ TestStorageMonitor::Destroy();
+ }
+
+ void StartDialog() {
+ ASSERT_FALSE(controller_);
+ controller_ = new MediaGalleriesDialogController(
+ *extension_.get(),
+ gallery_prefs_.get(),
+ base::Bind(&MediaGalleriesDialogControllerTest::CreateMockDialog,
+ base::Unretained(this)),
+ base::Bind(
+ &MediaGalleriesDialogControllerTest::OnControllerDone,
+ base::Unretained(this)));
+ }
+
+ MediaGalleriesDialogController* controller() {
+ return controller_;
+ }
+
+ MockMediaGalleriesDialog* dialog() {
+ return dialog_;
+ }
+
+ int dialog_update_count_at_destruction() {
+ EXPECT_FALSE(dialog_);
+ return dialog_update_count_at_destruction_;
+ }
+
+ extensions::Extension* extension() {
+ return extension_.get();
+ }
+
+ MediaGalleriesPreferences* gallery_prefs() {
+ return gallery_prefs_.get();
+ }
+
+ void TestForgottenType(MediaGalleryPrefInfo::Type type);
+
+ protected:
+ EnsureMediaDirectoriesExists mock_gallery_locations_;
+
+ private:
+ MediaGalleriesDialog* CreateMockDialog(
+ MediaGalleriesDialogController* controller) {
+ EXPECT_FALSE(dialog_);
+ dialog_update_count_at_destruction_ = 0;
+ dialog_ = new MockMediaGalleriesDialog(base::Bind(
+ &MediaGalleriesDialogControllerTest::OnDialogDestroyed,
+ weak_factory_.GetWeakPtr()));
+ return dialog_;
+ }
+
+ void OnDialogDestroyed(int update_count) {
+ EXPECT_TRUE(dialog_);
+ dialog_update_count_at_destruction_ = update_count;
+ dialog_ = NULL;
+ }
+
+ void OnControllerDone() {
+ controller_ = NULL;
+ }
+
+ // Needed for extension service & friends to work.
+ content::TestBrowserThreadBundle thread_bundle_;
+
+ // The dialog is owned by the controller, but this pointer should only be
+ // valid while the dialog is live within the controller.
+ MockMediaGalleriesDialog* dialog_;
+ int dialog_update_count_at_destruction_;
+
+ // The controller owns itself.
+ MediaGalleriesDialogController* controller_;
+
+ scoped_refptr<extensions::Extension> extension_;
+
+#if defined OS_CHROMEOS
+ chromeos::ScopedTestDeviceSettingsService test_device_settings_service_;
+ chromeos::ScopedTestCrosSettings test_cros_settings_;
+ chromeos::ScopedTestUserManager test_user_manager_;
+#endif
+
+ TestStorageMonitor monitor_;
+ scoped_ptr<TestingProfile> profile_;
+ scoped_ptr<MediaGalleriesPreferences> gallery_prefs_;
+
+ base::WeakPtrFactory<MediaGalleriesDialogControllerTest>
+ weak_factory_;
+
+ DISALLOW_COPY_AND_ASSIGN(MediaGalleriesDialogControllerTest);
+};
+
+void MediaGalleriesDialogControllerTest::TestForgottenType(
+ MediaGalleryPrefInfo::Type type) {
+ EXPECT_EQ(0U, gallery_prefs()->GalleriesForExtension(*extension()).size());
+
+ MediaGalleryPrefId forgotten1 = gallery_prefs()->AddGalleryByPath(
+ MakeMediaGalleriesTestingPath("forgotten1"), type);
+ MediaGalleryPrefId forgotten2 = gallery_prefs()->AddGalleryByPath(
+ MakeMediaGalleriesTestingPath("forgotten2"), type);
+ // Show dialog and accept to verify 2 entries
+ StartDialog();
+ EXPECT_EQ(mock_gallery_locations_.num_galleries() + 2U,
+ controller()->AttachedPermissions().size());
+ EXPECT_EQ(0U, controller()->UnattachedPermissions().size());
+ controller()->DidToggleGalleryId(forgotten1, true);
+ controller()->DidToggleGalleryId(forgotten2, true);
+ controller()->DialogFinished(true);
+ EXPECT_EQ(2U, gallery_prefs()->GalleriesForExtension(*extension()).size());
+
+ // Forget one and cancel to see that it's still there.
+ StartDialog();
+ controller()->DidForgetGallery(forgotten1);
+ EXPECT_EQ(mock_gallery_locations_.num_galleries() + 1U,
+ controller()->AttachedPermissions().size());
+ controller()->DialogFinished(false);
+ EXPECT_EQ(2U, gallery_prefs()->GalleriesForExtension(*extension()).size());
+
+ // Forget one and confirm to see that it's gone.
+ StartDialog();
+ controller()->DidForgetGallery(forgotten1);
+ EXPECT_EQ(mock_gallery_locations_.num_galleries() + 1U,
+ controller()->AttachedPermissions().size());
+ controller()->DialogFinished(true);
+ EXPECT_EQ(1U, gallery_prefs()->GalleriesForExtension(*extension()).size());
+
+ // Add a new one and forget it & see that it's gone.
+ MediaGalleryPrefId forgotten3 = gallery_prefs()->AddGalleryByPath(
+ MakeMediaGalleriesTestingPath("forgotten3"), type);
+ StartDialog();
+ EXPECT_EQ(mock_gallery_locations_.num_galleries() + 2U,
+ controller()->AttachedPermissions().size());
+ EXPECT_EQ(0U, controller()->UnattachedPermissions().size());
+ controller()->DidToggleGalleryId(forgotten3, true);
+ controller()->DidForgetGallery(forgotten3);
+ EXPECT_EQ(mock_gallery_locations_.num_galleries() + 1U,
+ controller()->AttachedPermissions().size());
+ controller()->DialogFinished(true);
+ EXPECT_EQ(1U, gallery_prefs()->GalleriesForExtension(*extension()).size());
+}
+
+TEST_F(MediaGalleriesDialogControllerTest, TestForgottenUserAdded) {
+ TestForgottenType(MediaGalleryPrefInfo::kUserAdded);
+}
+
+TEST_F(MediaGalleriesDialogControllerTest, TestForgottenAutoDetected) {
+ TestForgottenType(MediaGalleryPrefInfo::kAutoDetected);
+}
+
+TEST_F(MediaGalleriesDialogControllerTest, TestForgottenScanResult) {
+ TestForgottenType(MediaGalleryPrefInfo::kScanResult);
+}
+
+TEST_F(MediaGalleriesDialogControllerTest, TestNameGeneration) {