// 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. #include #include #include #include "base/json/json_writer.h" #include "base/memory/scoped_ptr.h" #include "base/stringprintf.h" #include "base/utf_string_conversions.h" #include "base/values.h" #include "chrome/app/chrome_command_ids.h" #include "chrome/browser/infobars/infobar.h" #include "chrome/browser/infobars/infobar_tab_helper.h" #include "chrome/browser/prefs/pref_change_registrar.h" #include "chrome/browser/prefs/pref_service.h" #include "chrome/browser/tab_contents/render_view_context_menu.h" #include "chrome/browser/translate/translate_infobar_delegate.h" #include "chrome/browser/translate/translate_manager.h" #include "chrome/browser/translate/translate_prefs.h" #include "chrome/browser/ui/tab_contents/tab_contents_wrapper.h" #include "chrome/browser/ui/tab_contents/test_tab_contents_wrapper.h" #include "chrome/common/chrome_notification_types.h" #include "chrome/common/pref_names.h" #include "chrome/common/render_messages.h" #include "chrome/test/base/testing_browser_process.h" #include "chrome/test/base/testing_profile.h" #include "content/browser/renderer_host/mock_render_process_host.h" #include "content/browser/tab_contents/test_tab_contents.h" #include "content/public/browser/navigation_details.h" #include "content/public/browser/navigation_entry.h" #include "content/public/browser/notification_details.h" #include "content/public/browser/notification_registrar.h" #include "content/test/notification_observer_mock.h" #include "content/test/test_browser_thread.h" #include "content/test/test_url_fetcher_factory.h" #include "grit/generated_resources.h" #include "ipc/ipc_test_sink.h" #include "testing/gmock/include/gmock/gmock.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebContextMenuData.h" #include "third_party/cld/languages/public/languages.h" #if defined(USE_WEBKIT_COMPOSITOR) #include "content/test/render_view_test.h" #include "third_party/WebKit/Source/WebKit/chromium/public/WebKit.h" #endif using content::BrowserThread; using content::NavigationController; using content::WebContents; using testing::_; using testing::Pointee; using testing::Property; using WebKit::WebContextMenuData; class TranslateManagerTest : public TabContentsWrapperTestHarness, public content::NotificationObserver { public: TranslateManagerTest() : ui_thread_(BrowserThread::UI, &message_loop_) { } // Simulates navigating to a page and getting the page contents and language // for that navigation. void SimulateNavigation(const GURL& url, const std::string& lang, bool page_translatable) { NavigateAndCommit(url); SimulateOnTranslateLanguageDetermined(lang, page_translatable); } void SimulateOnTranslateLanguageDetermined(const std::string& lang, bool page_translatable) { rvh()->TestOnMessageReceived(ChromeViewHostMsg_TranslateLanguageDetermined( 0, lang, page_translatable)); } bool GetTranslateMessage(int* page_id, std::string* original_lang, std::string* target_lang) { const IPC::Message* message = process()->sink().GetFirstMessageMatching( ChromeViewMsg_TranslatePage::ID); if (!message) return false; Tuple4 translate_param; ChromeViewMsg_TranslatePage::Read(message, &translate_param); if (page_id) *page_id = translate_param.a; // Ignore translate_param.b which is the script injected in the page. if (original_lang) *original_lang = translate_param.c; if (target_lang) *target_lang = translate_param.d; return true; } InfoBarTabHelper* infobar_tab_helper() { return contents_wrapper()->infobar_tab_helper(); } // Returns the translate infobar if there is 1 infobar and it is a translate // infobar. TranslateInfoBarDelegate* GetTranslateInfoBar() { return (infobar_tab_helper()->infobar_count() == 1) ? infobar_tab_helper()->GetInfoBarDelegateAt(0)-> AsTranslateInfoBarDelegate() : NULL; } // If there is 1 infobar and it is a translate infobar, closes it and returns // true. Returns false otherwise. bool CloseTranslateInfoBar() { InfoBarDelegate* infobar = GetTranslateInfoBar(); if (!infobar) return false; infobar->InfoBarDismissed(); // Simulates closing the infobar. infobar_tab_helper()->RemoveInfoBar(infobar); return true; } // Checks whether |infobar| has been removed and clears the removed infobar // list. bool CheckInfoBarRemovedAndReset(InfoBarDelegate* delegate) { bool found = removed_infobars_.count(delegate) != 0; removed_infobars_.clear(); return found; } // Returns true if at least one infobar was closed. bool InfoBarRemoved() { return !removed_infobars_.empty(); } // Clears the list of stored removed infobars. void ClearRemovedInfoBars() { removed_infobars_.clear(); } void ExpireTranslateScriptImmediately() { TranslateManager::GetInstance()->set_translate_script_expiration_delay(0); } // If there is 1 infobar and it is a translate infobar, deny translation and // returns true. Returns false otherwise. bool DenyTranslation() { TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); if (!infobar) return false; infobar->TranslationDeclined(); infobar_tab_helper()->RemoveInfoBar(infobar); return true; } virtual void Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) { DCHECK_EQ(chrome::NOTIFICATION_TAB_CONTENTS_INFOBAR_REMOVED, type); removed_infobars_.insert( content::Details(details)->first); } protected: virtual void SetUp() { #if defined(USE_WEBKIT_COMPOSITOR) WebKit::initialize(&webkit_platform_support_); #endif // Access the TranslateManager singleton so it is created before we call // TabContentsWrapperTestHarness::SetUp() to match what's done in Chrome, // where the TranslateManager is created before the TabContents. This // matters as they both register for similar events and we want the // notifications to happen in the same sequence (TranslateManager first, // TabContents second). Also clears the translate script so it is fetched // everytime and sets the expiration delay to a large value by default (in // case it was zeroed in a previous test). TranslateManager::GetInstance()->ClearTranslateScript(); TranslateManager::GetInstance()-> set_translate_script_expiration_delay(60 * 60 * 1000); TabContentsWrapperTestHarness::SetUp(); notification_registrar_.Add(this, chrome::NOTIFICATION_TAB_CONTENTS_INFOBAR_REMOVED, content::Source( contents_wrapper()->infobar_tab_helper())); } virtual void TearDown() { process()->sink().ClearMessages(); notification_registrar_.Remove(this, chrome::NOTIFICATION_TAB_CONTENTS_INFOBAR_REMOVED, content::Source( contents_wrapper()->infobar_tab_helper())); TabContentsWrapperTestHarness::TearDown(); #if defined(USE_WEBKIT_COMPOSITOR) WebKit::shutdown(); #endif } void SimulateTranslateScriptURLFetch(bool success) { TestURLFetcher* fetcher = url_fetcher_factory_.GetFetcherByID(0); ASSERT_TRUE(fetcher); net::URLRequestStatus status; status.set_status(success ? net::URLRequestStatus::SUCCESS : net::URLRequestStatus::FAILED); fetcher->set_url(fetcher->GetOriginalURL()); fetcher->set_status(status); fetcher->set_response_code(success ? 200 : 500); fetcher->delegate()->OnURLFetchComplete(fetcher); } void SimulateSupportedLanguagesURLFetch( bool success, const std::vector& languages) { TestURLFetcher* fetcher = url_fetcher_factory_.GetFetcherByID(1); ASSERT_TRUE(fetcher); net::URLRequestStatus status; status.set_status(success ? net::URLRequestStatus::SUCCESS : net::URLRequestStatus::FAILED); std::string data; if (success) { data = base::StringPrintf("%s{'sl': {'bla': 'bla'}, '%s': {", TranslateManager::kLanguageListCallbackName, TranslateManager::kTargetLanguagesKey); const char* comma = ""; for (size_t i = 0; i < languages.size(); ++i) { data += base::StringPrintf( "%s'%s': 'UnusedFullName'", comma, languages[i].c_str()); if (i == 0) comma = ","; } data += "}})"; } fetcher->set_url(fetcher->GetOriginalURL()); fetcher->set_status(status); fetcher->set_response_code(success ? 200 : 500); fetcher->SetResponseString(data); fetcher->delegate()->OnURLFetchComplete(fetcher); } void SetPrefObserverExpectation(const char* path) { EXPECT_CALL( pref_observer_, Observe(int(chrome::NOTIFICATION_PREF_CHANGED), _, Property(&content::Details::ptr, Pointee(path)))); } content::NotificationObserverMock pref_observer_; private: content::NotificationRegistrar notification_registrar_; TestURLFetcherFactory url_fetcher_factory_; content::TestBrowserThread ui_thread_; #if defined(USE_WEBKIT_COMPOSITOR) content::RenderViewTest::RendererWebKitPlatformSupportImplNoSandbox webkit_platform_support_; #endif // The infobars that have been removed. // WARNING: the pointers point to deleted objects, use only for comparison. std::set removed_infobars_; DISALLOW_COPY_AND_ASSIGN(TranslateManagerTest); }; // An observer that keeps track of whether a navigation entry was committed. class NavEntryCommittedObserver : public content::NotificationObserver { public: explicit NavEntryCommittedObserver(TabContents* tab_contents) { registrar_.Add(this, content::NOTIFICATION_NAV_ENTRY_COMMITTED, content::Source( &tab_contents->GetController())); } virtual void Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) { DCHECK(type == content::NOTIFICATION_NAV_ENTRY_COMMITTED); details_ = *(content::Details(details).ptr()); } const content::LoadCommittedDetails& get_load_commited_details() const { return details_; } private: content::LoadCommittedDetails details_; content::NotificationRegistrar registrar_; DISALLOW_COPY_AND_ASSIGN(NavEntryCommittedObserver); }; namespace { class TestRenderViewContextMenu : public RenderViewContextMenu { public: static TestRenderViewContextMenu* CreateContextMenu( WebContents* web_contents) { ContextMenuParams params; params.media_type = WebKit::WebContextMenuData::MediaTypeNone; params.x = 0; params.y = 0; params.is_image_blocked = false; params.media_flags = 0; params.spellcheck_enabled = false; params.is_editable = false; params.page_url = web_contents->GetController().GetActiveEntry()->GetURL(); #if defined(OS_MACOSX) params.writing_direction_default = 0; params.writing_direction_left_to_right = 0; params.writing_direction_right_to_left = 0; #endif // OS_MACOSX params.edit_flags = WebContextMenuData::CanTranslate; return new TestRenderViewContextMenu(web_contents, params); } bool IsItemPresent(int id) { return menu_model_.GetIndexOfCommandId(id) != -1; } virtual void PlatformInit() { } virtual bool GetAcceleratorForCommandId( int command_id, ui::Accelerator* accelerator) { return false; } private: TestRenderViewContextMenu(WebContents* web_contents, const ContextMenuParams& params) : RenderViewContextMenu(web_contents, params) { } DISALLOW_COPY_AND_ASSIGN(TestRenderViewContextMenu); }; } // namespace TEST_F(TranslateManagerTest, NormalTranslate) { // Simulate navigating to a page. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // We should have an infobar. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::BEFORE_TRANSLATE, infobar->type()); // Simulate clicking translate. process()->sink().ClearMessages(); infobar->Translate(); // The "Translating..." infobar should be showing. infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATING, infobar->type()); // Simulate the translate script being retrieved (it only needs to be done // once in the test as it is cached). SimulateTranslateScriptURLFetch(true); // Test that we sent the right message to the renderer. int page_id = 0; std::string original_lang, target_lang; EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ("fr", original_lang); EXPECT_EQ("en", target_lang); // Simulate the render notifying the translation has been done. rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // The after translate infobar should be showing. infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::AFTER_TRANSLATE, infobar->type()); // Simulate changing the original language, this should trigger a translation. process()->sink().ClearMessages(); std::string new_original_lang = infobar->GetLanguageCodeAt(0); infobar->SetOriginalLanguage(0); EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ(new_original_lang, original_lang); EXPECT_EQ("en", target_lang); // Simulate the render notifying the translation has been done. rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, new_original_lang, "en", TranslateErrors::NONE)); // infobar is now invalid. TranslateInfoBarDelegate* new_infobar = GetTranslateInfoBar(); ASSERT_TRUE(new_infobar != NULL); infobar = new_infobar; // Simulate changing the target language, this should trigger a translation. process()->sink().ClearMessages(); std::string new_target_lang = infobar->GetLanguageCodeAt(1); infobar->SetTargetLanguage(1); EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ(new_original_lang, original_lang); EXPECT_EQ(new_target_lang, target_lang); // Simulate the render notifying the translation has been done. rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated(0, 0, new_original_lang, new_target_lang, TranslateErrors::NONE)); // infobar is now invalid. new_infobar = GetTranslateInfoBar(); ASSERT_TRUE(new_infobar != NULL); } TEST_F(TranslateManagerTest, TranslateScriptNotAvailable) { // Simulate navigating to a page. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // We should have an infobar. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::BEFORE_TRANSLATE, infobar->type()); // Simulate clicking translate. process()->sink().ClearMessages(); infobar->Translate(); // Simulate a failure retrieving the translate script. SimulateTranslateScriptURLFetch(false); // We should not have sent any message to translate to the renderer. EXPECT_FALSE(GetTranslateMessage(NULL, NULL, NULL)); // And we should have an error infobar showing. infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATION_ERROR, infobar->type()); } // Ensures we deal correctly with pages for which the browser does not recognize // the language (the translate server may or not detect the language). TEST_F(TranslateManagerTest, TranslateUnknownLanguage) { // Simulate navigating to a page ("und" is the string returned by the CLD for // languages it does not recognize). SimulateNavigation(GURL("http://www.google.mys"), "und", true); // We should not have an infobar as we don't know the language. ASSERT_TRUE(GetTranslateInfoBar() == NULL); // Translate the page anyway throught the context menu. scoped_ptr menu( TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); menu->ExecuteCommand(IDC_CONTENT_CONTEXT_TRANSLATE); // To test that bug #49018 if fixed, make sure we deal correctly with errors. // Simulate a failure to fetch the translate script. SimulateTranslateScriptURLFetch(false); TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATION_ERROR, infobar->type()); EXPECT_TRUE(infobar->IsError()); infobar->MessageInfoBarButtonPressed(); SimulateTranslateScriptURLFetch(true); // This time succeed. // Simulate the render notifying the translation has been done, the server // having detected the page was in a known and supported language. rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // The after translate infobar should be showing. infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::AFTER_TRANSLATE, infobar->type()); EXPECT_EQ("fr", infobar->GetOriginalLanguageCode()); EXPECT_EQ("en", infobar->GetTargetLanguageCode()); // Let's run the same steps but this time the server detects the page is // already in English. SimulateNavigation(GURL("http://www.google.com"), "und", true); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); menu->ExecuteCommand(IDC_CONTENT_CONTEXT_TRANSLATE); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 1, 0, "en", "en", TranslateErrors::IDENTICAL_LANGUAGES)); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATION_ERROR, infobar->type()); EXPECT_EQ(TranslateErrors::IDENTICAL_LANGUAGES, infobar->error()); // Let's run the same steps again but this time the server fails to detect the // page's language (it returns an empty string). SimulateNavigation(GURL("http://www.google.com"), "und", true); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); menu->ExecuteCommand(IDC_CONTENT_CONTEXT_TRANSLATE); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 2, 0, "", "en", TranslateErrors::UNKNOWN_LANGUAGE)); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATION_ERROR, infobar->type()); EXPECT_EQ(TranslateErrors::UNKNOWN_LANGUAGE, infobar->error()); } // Tests that we show/don't show an info-bar for all languages the CLD can // report. TEST_F(TranslateManagerTest, TestAllLanguages) { // The index in kExpectation are the Language enum (see languages.pb.h). // true if we expect a translate infobar for that language. // Note the supported languages are in translation_manager.cc, see // kSupportedLanguages. bool kExpectations[] = { // 0-9 false, true, true, true, true, true, true, true, true, true, // 10-19 true, true, true, true, true, true, true, true, true, true, // 20-29 true, true, true, true, true, false, false, true, true, true, // 30-39 true, true, true, true, true, true, true, false, true, false, // 40-49 true, false, true, false, false, true, false, true, false, false, // 50-59 true, false, false, true, true, true, false, true, false, false, // 60-69 false, false, true, true, false, true, true, false, true, true, // 70-79 false, false, false, false, true, true, false, true, false, false, // 80-89 false, true, true, false, false, false, false, false, false, false, // 90-99 false, true, false, false, false, false, false, true, false, false, // 100-109 false, true, false, false, false, false, false, false, false, false, // 110-119 false, false, false, false, false, false, false, false, false, false, // 120-129 false, false, false, false, false, false, false, false, false, false, // 130-139 false, false, false, false, false, false, false, false, false, true, // 140-149 false, false, false, false, false, false, false, false, false, false, // 150-159 false, false, false, false, false, false, false, false, false, false, // 160 true }; GURL url("http://www.google.com"); for (size_t i = 0; i < arraysize(kExpectations); ++i) { ASSERT_LT(i, static_cast(NUM_LANGUAGES)); std::string lang = LanguageCodeWithDialects(static_cast(i)); SCOPED_TRACE(::testing::Message() << "Iteration " << i << " language=" << lang); // We should not have a translate infobar. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar == NULL); // Simulate navigating to a page. NavigateAndCommit(url); SimulateOnTranslateLanguageDetermined(lang, true); // Verify we have/don't have an info-bar as expected. infobar = GetTranslateInfoBar(); EXPECT_EQ(kExpectations[i], infobar != NULL); // Close the info-bar if applicable. if (infobar != NULL) EXPECT_TRUE(CloseTranslateInfoBar()); } } // Test the fetching of languages from the translate server TEST_F(TranslateManagerTest, FetchLanguagesFromTranslateServer) { std::vector server_languages; // A list of languages to fake being returned by the translate server. server_languages.push_back("aa"); server_languages.push_back("bb"); server_languages.push_back("ab"); server_languages.push_back("en-CA"); server_languages.push_back("zz"); server_languages.push_back("yy"); server_languages.push_back("fr-FR"); // First, get the default languages list: std::vector default_supported_languages; TranslateManager::GetSupportedLanguages(&default_supported_languages); // To make sure we got the defaults and don't confuse them with the mocks. ASSERT_NE(default_supported_languages.size(), server_languages.size()); Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); TranslateManager::GetInstance()->FetchLanguageListFromTranslateServer(prefs); // Check that we still get the defaults until the URLFetch has completed. std::vector current_supported_languages; TranslateManager::GetSupportedLanguages(¤t_supported_languages); EXPECT_EQ(default_supported_languages, current_supported_languages); // Also check that it didn't change if we failed the URL fetch. SimulateSupportedLanguagesURLFetch(false, std::vector()); current_supported_languages.clear(); TranslateManager::GetSupportedLanguages(¤t_supported_languages); EXPECT_EQ(default_supported_languages, current_supported_languages); // Now check that we got the appropriate set of languages from the server. TranslateManager::GetInstance()->FetchLanguageListFromTranslateServer(prefs); SimulateSupportedLanguagesURLFetch(true, server_languages); current_supported_languages.clear(); TranslateManager::GetSupportedLanguages(¤t_supported_languages); // Not sure we need to guarantee the order of languages, so we find them. EXPECT_EQ(server_languages.size(), current_supported_languages.size()); for (size_t i = 0; i < server_languages.size(); ++i) { EXPECT_NE(current_supported_languages.end(), std::find(current_supported_languages.begin(), current_supported_languages.end(), server_languages[i])); } // Reset to original state. TranslateManager::GetInstance()->FetchLanguageListFromTranslateServer(prefs); SimulateSupportedLanguagesURLFetch(true, default_supported_languages); } std::string GetLanguageListString( const std::vector& language_list) { // The translate manager is expecting a JSON string like: // sl({'sl': {'XX': 'LanguageName', ...}, 'tl': {'XX': 'LanguageName', ...}}) // We only need to set the tl (target languages) dictionary. scoped_ptr target_languages_dict(new DictionaryValue); for (size_t i = 0; i < language_list.size(); ++i) { // The value is ignored, we only use the key. target_languages_dict->Set(language_list[i], Value::CreateNullValue()); } DictionaryValue language_list_dict; language_list_dict.Set("tl", target_languages_dict.release()); std::string language_list_json_str; base::JSONWriter::Write(&language_list_dict, false, &language_list_json_str); std::string language_list_str("sl("); language_list_str += language_list_json_str; language_list_str += ")"; return language_list_str; } // Test Language Code synonyms. TEST_F(TranslateManagerTest, LanguageCodeSynonyms) { // The current set of synonyms are {"nb", "no"}, {"he", "iw"}, {"jw", "jv"}. std::vector language_list; // Add some values around ht potential synonyms. language_list.push_back("fr"); language_list.push_back("nb"); language_list.push_back("en"); TranslateManager::SetSupportedLanguages(GetLanguageListString(language_list)); EXPECT_TRUE(TranslateManager::IsSupportedLanguage("no")); EXPECT_TRUE(TranslateManager::IsSupportedLanguage("nb")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("he")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("iw")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("jw")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("jv")); language_list.push_back("iw"); TranslateManager::SetSupportedLanguages(GetLanguageListString(language_list)); EXPECT_TRUE(TranslateManager::IsSupportedLanguage("he")); EXPECT_TRUE(TranslateManager::IsSupportedLanguage("iw")); language_list.clear(); language_list.push_back("jw"); TranslateManager::SetSupportedLanguages(GetLanguageListString(language_list)); EXPECT_TRUE(TranslateManager::IsSupportedLanguage("jw")); EXPECT_TRUE(TranslateManager::IsSupportedLanguage("jv")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("no")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("nb")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("he")); EXPECT_FALSE(TranslateManager::IsSupportedLanguage("iw")); } // Tests auto-translate on page. TEST_F(TranslateManagerTest, AutoTranslateOnNavigate) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Simulate the user translating. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); infobar->Translate(); // Simulate the translate script being retrieved. SimulateTranslateScriptURLFetch(true); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // Now navigate to a new page in the same language. process()->sink().ClearMessages(); SimulateNavigation(GURL("http://news.google.fr"), "fr", true); // This should have automatically triggered a translation. int page_id = 0; std::string original_lang, target_lang; EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ(1, page_id); EXPECT_EQ("fr", original_lang); EXPECT_EQ("en", target_lang); // Now navigate to a page in a different language. process()->sink().ClearMessages(); SimulateNavigation(GURL("http://news.google.es"), "es", true); // This should not have triggered a translate. EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); } // Tests that multiple OnPageContents do not cause multiple infobars. TEST_F(TranslateManagerTest, MultipleOnPageContents) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Simulate clicking 'Nope' (don't translate). EXPECT_TRUE(DenyTranslation()); EXPECT_EQ(0U, infobar_tab_helper()->infobar_count()); // Send a new PageContents, we should not show an infobar. SimulateOnTranslateLanguageDetermined("fr", true); EXPECT_EQ(0U, infobar_tab_helper()->infobar_count()); // Do the same steps but simulate closing the infobar this time. SimulateNavigation(GURL("http://www.youtube.fr"), "fr", true); EXPECT_TRUE(CloseTranslateInfoBar()); EXPECT_EQ(0U, infobar_tab_helper()->infobar_count()); SimulateOnTranslateLanguageDetermined("fr", true); EXPECT_EQ(0U, infobar_tab_helper()->infobar_count()); } // Test that reloading the page brings back the infobar. TEST_F(TranslateManagerTest, Reload) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Reload should bring back the infobar. NavEntryCommittedObserver nav_observer(contents()); Reload(); // Ensures it is really handled a reload. const content::LoadCommittedDetails& nav_details = nav_observer.get_load_commited_details(); EXPECT_TRUE(nav_details.entry != NULL); // There was a navigation. EXPECT_EQ(content::NAVIGATION_TYPE_EXISTING_PAGE, nav_details.type); // The TranslateManager class processes the navigation entry committed // notification in a posted task; process that task. MessageLoop::current()->RunAllPending(); EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Test that reloading the page by way of typing again the URL in the // location bar brings back the infobar. TEST_F(TranslateManagerTest, ReloadFromLocationBar) { GURL url("http://www.google.fr"); // Simulate navigating to a page and getting its language. SimulateNavigation(url, "fr", true); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Create a pending navigation and simulate a page load. That should be the // equivalent of typing the URL again in the location bar. NavEntryCommittedObserver nav_observer(contents()); contents()->GetController().LoadURL(url, content::Referrer(), content::PAGE_TRANSITION_TYPED, std::string()); rvh()->SendNavigate(0, url); // Test that we are really getting a same page navigation, the test would be // useless if it was not the case. const content::LoadCommittedDetails& nav_details = nav_observer.get_load_commited_details(); EXPECT_TRUE(nav_details.entry != NULL); // There was a navigation. EXPECT_EQ(content::NAVIGATION_TYPE_SAME_PAGE, nav_details.type); // The TranslateManager class processes the navigation entry committed // notification in a posted task; process that task. MessageLoop::current()->RunAllPending(); EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that a closed translate infobar does not reappear when navigating // in-page. TEST_F(TranslateManagerTest, CloseInfoBarInPageNavigation) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Navigate in page, no infobar should be shown. SimulateNavigation(GURL("http://www.google.fr/#ref1"), "fr", true); EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Navigate out of page, a new infobar should show. SimulateNavigation(GURL("http://www.google.fr/foot"), "fr", true); EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that a closed translate infobar does not reappear when navigating // in a subframe. (http://crbug.com/48215) TEST_F(TranslateManagerTest, CloseInfoBarInSubframeNavigation) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Simulate a sub-frame auto-navigating. rvh()->SendNavigateWithTransition(1, GURL("http://pub.com"), content::PAGE_TRANSITION_AUTO_SUBFRAME); EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Simulate the user navigating in a sub-frame. rvh()->SendNavigateWithTransition(2, GURL("http://pub.com"), content::PAGE_TRANSITION_MANUAL_SUBFRAME); EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Navigate out of page, a new infobar should show. SimulateNavigation(GURL("http://www.google.fr/foot"), "fr", true); EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that denying translation is sticky when navigating in page. TEST_F(TranslateManagerTest, DenyTranslateInPageNavigation) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Simulate clicking 'Nope' (don't translate). EXPECT_TRUE(DenyTranslation()); // Navigate in page, no infobar should be shown. SimulateNavigation(GURL("http://www.google.fr/#ref1"), "fr", true); EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Navigate out of page, a new infobar should show. SimulateNavigation(GURL("http://www.google.fr/foot"), "fr", true); EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that after translating and closing the infobar, the infobar does not // return when navigating in page. TEST_F(TranslateManagerTest, TranslateCloseInfoBarInPageNavigation) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Simulate the user translating. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); infobar->Translate(); // Simulate the translate script being retrieved. SimulateTranslateScriptURLFetch(true); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Navigate in page, no infobar should be shown. SimulateNavigation(GURL("http://www.google.fr/#ref1"), "fr", true); EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Navigate out of page, a new infobar should show. // Note that we navigate to a page in a different language so we don't trigger // the auto-translate feature (it would translate the page automatically and // the before translate inforbar would not be shown). SimulateNavigation(GURL("http://www.google.de"), "de", true); EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that the after translate the infobar still shows when navigating // in-page. TEST_F(TranslateManagerTest, TranslateInPageNavigation) { // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // Simulate the user translating. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); infobar->Translate(); // Simulate the translate script being retrieved. SimulateTranslateScriptURLFetch(true); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // The after translate infobar is showing. infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); // Navigate out of page, a new infobar should show. // See note in TranslateCloseInfoBarInPageNavigation test on why it is // important to navigate to a page in a different language for this test. SimulateNavigation(GURL("http://www.google.de"), "de", true); // The old infobar is gone. EXPECT_TRUE(CheckInfoBarRemovedAndReset(infobar)); // And there is a new one. EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that no translate infobar is shown when navigating to a page in an // unsupported language. TEST_F(TranslateManagerTest, CLDReportsUnsupportedPageLanguage) { // Simulate navigating to a page and getting an unsupported language. SimulateNavigation(GURL("http://www.google.com"), "qbz", true); // No info-bar should be shown. EXPECT_TRUE(GetTranslateInfoBar() == NULL); } // Tests that we deal correctly with unsupported languages returned by the // server. // The translation server might return a language we don't support. TEST_F(TranslateManagerTest, ServerReportsUnsupportedLanguage) { // Simulate navigating to a page and translating it. SimulateNavigation(GURL("http://mail.google.fr"), "fr", true); TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); process()->sink().ClearMessages(); infobar->Translate(); SimulateTranslateScriptURLFetch(true); // Simulate the render notifying the translation has been done, but it // reports a language we don't support. rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "qbz", "en", TranslateErrors::NONE)); // An error infobar should be showing to report that we don't support this // language. infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATION_ERROR, infobar->type()); // This infobar should have a button (so the string should not be empty). ASSERT_FALSE(infobar->GetMessageInfoBarButtonText().empty()); // Pressing the button on that infobar should revert to the original language. process()->sink().ClearMessages(); infobar->MessageInfoBarButtonPressed(); const IPC::Message* message = process()->sink().GetFirstMessageMatching( ChromeViewMsg_RevertTranslation::ID); EXPECT_TRUE(message != NULL); // And it should have removed the infobar. EXPECT_TRUE(GetTranslateInfoBar() == NULL); } // Tests that no translate infobar is shown when Chrome is in a language that // the translate server does not support. TEST_F(TranslateManagerTest, UnsupportedUILanguage) { std::string original_lang = g_browser_process->GetApplicationLocale(); g_browser_process->SetApplicationLocale("qbz"); // Make sure that the accept language list only contains unsupported languages Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); prefs->SetString(prefs::kAcceptLanguages, "qbz"); // Simulate navigating to a page in a language supported by the translate // server. SimulateNavigation(GURL("http://www.google.com"), "en", true); // No info-bar should be shown. EXPECT_TRUE(GetTranslateInfoBar() == NULL); g_browser_process->SetApplicationLocale(original_lang); } // Tests that the first supported accept language is selected TEST_F(TranslateManagerTest, TranslateAcceptLanguage) { // Set locate to non-existant language std::string original_lang = g_browser_process->GetApplicationLocale(); g_browser_process->SetApplicationLocale("qbz"); // Set Qbz and French as the only accepted languages Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); prefs->SetString(prefs::kAcceptLanguages, "qbz,fr"); // Go to a German page SimulateNavigation(GURL("http://google.de"), "de", true); // Expect the infobar to pop up EXPECT_TRUE(GetTranslateInfoBar() != NULL); // Set Qbz and English-US as the only accepted languages to test the country // code removal code which was causing a crash as filed in Issue 90106, // a crash caused by a language with a country code that wasn't recognized. prefs->SetString(prefs::kAcceptLanguages, "qbz,en-us"); // Go to a German page SimulateNavigation(GURL("http://google.de"), "de", true); // Expect the infobar to pop up EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests that the translate enabled preference is honored. TEST_F(TranslateManagerTest, TranslateEnabledPref) { // Make sure the pref allows translate. Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); prefs->SetBoolean(prefs::kEnableTranslate, true); // Simulate navigating to a page and getting its language. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // An infobar should be shown. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); EXPECT_TRUE(infobar != NULL); // Disable translate. prefs->SetBoolean(prefs::kEnableTranslate, false); // Navigate to a new page, that should close the previous infobar. GURL url("http://www.youtube.fr"); NavigateAndCommit(url); infobar = GetTranslateInfoBar(); EXPECT_TRUE(infobar == NULL); // Simulate getting the page contents and language, that should not trigger // a translate infobar. SimulateOnTranslateLanguageDetermined("fr", true); infobar = GetTranslateInfoBar(); EXPECT_TRUE(infobar == NULL); } // Tests the "Never translate " pref. TEST_F(TranslateManagerTest, NeverTranslateLanguagePref) { // Simulate navigating to a page and getting its language. GURL url("http://www.google.fr"); SimulateNavigation(url, "fr", true); // An infobar should be shown. EXPECT_TRUE(GetTranslateInfoBar() != NULL); // Select never translate this language. Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); PrefChangeRegistrar registrar; registrar.Init(prefs); registrar.Add(TranslatePrefs::kPrefTranslateLanguageBlacklist, &pref_observer_); TranslatePrefs translate_prefs(prefs); EXPECT_FALSE(translate_prefs.IsLanguageBlacklisted("fr")); EXPECT_TRUE(translate_prefs.CanTranslate(prefs, "fr", url)); SetPrefObserverExpectation(TranslatePrefs::kPrefTranslateLanguageBlacklist); translate_prefs.BlacklistLanguage("fr"); EXPECT_TRUE(translate_prefs.IsLanguageBlacklisted("fr")); EXPECT_FALSE(translate_prefs.CanTranslate(prefs, "fr", url)); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Navigate to a new page also in French. SimulateNavigation(GURL("http://wwww.youtube.fr"), "fr", true); // There should not be a translate infobar. EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Remove the language from the blacklist. SetPrefObserverExpectation(TranslatePrefs::kPrefTranslateLanguageBlacklist); translate_prefs.RemoveLanguageFromBlacklist("fr"); EXPECT_FALSE(translate_prefs.IsLanguageBlacklisted("fr")); EXPECT_TRUE(translate_prefs.CanTranslate(prefs, "fr", url)); // Navigate to a page in French. SimulateNavigation(url, "fr", true); // There should be a translate infobar. EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests the "Never translate this site" pref. TEST_F(TranslateManagerTest, NeverTranslateSitePref) { // Simulate navigating to a page and getting its language. GURL url("http://www.google.fr"); std::string host(url.host()); SimulateNavigation(url, "fr", true); // An infobar should be shown. EXPECT_TRUE(GetTranslateInfoBar() != NULL); // Select never translate this site. Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); PrefChangeRegistrar registrar; registrar.Init(prefs); registrar.Add(TranslatePrefs::kPrefTranslateSiteBlacklist, &pref_observer_); TranslatePrefs translate_prefs(prefs); EXPECT_FALSE(translate_prefs.IsSiteBlacklisted(host)); EXPECT_TRUE(translate_prefs.CanTranslate(prefs, "fr", url)); SetPrefObserverExpectation(TranslatePrefs::kPrefTranslateSiteBlacklist); translate_prefs.BlacklistSite(host); EXPECT_TRUE(translate_prefs.IsSiteBlacklisted(host)); EXPECT_FALSE(translate_prefs.CanTranslate(prefs, "fr", url)); // Close the infobar. EXPECT_TRUE(CloseTranslateInfoBar()); // Navigate to a new page also on the same site. SimulateNavigation(GURL("http://www.google.fr/hello"), "fr", true); // There should not be a translate infobar. EXPECT_TRUE(GetTranslateInfoBar() == NULL); // Remove the site from the blacklist. SetPrefObserverExpectation(TranslatePrefs::kPrefTranslateSiteBlacklist); translate_prefs.RemoveSiteFromBlacklist(host); EXPECT_FALSE(translate_prefs.IsSiteBlacklisted(host)); EXPECT_TRUE(translate_prefs.CanTranslate(prefs, "fr", url)); // Navigate to a page in French. SimulateNavigation(url, "fr", true); // There should be a translate infobar. EXPECT_TRUE(GetTranslateInfoBar() != NULL); } // Tests the "Always translate this language" pref. TEST_F(TranslateManagerTest, AlwaysTranslateLanguagePref) { // Select always translate French to English. Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); PrefService* prefs = profile->GetPrefs(); PrefChangeRegistrar registrar; registrar.Init(prefs); registrar.Add(TranslatePrefs::kPrefTranslateWhitelists, &pref_observer_); TranslatePrefs translate_prefs(prefs); SetPrefObserverExpectation(TranslatePrefs::kPrefTranslateWhitelists); translate_prefs.WhitelistLanguagePair("fr", "en"); // Load a page in French. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); // It should have triggered an automatic translation to English. // The translating infobar should be showing. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATING, infobar->type()); // Simulate the translate script being retrieved. SimulateTranslateScriptURLFetch(true); int page_id = 0; std::string original_lang, target_lang; EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ("fr", original_lang); EXPECT_EQ("en", target_lang); process()->sink().ClearMessages(); // Try another language, it should not be autotranslated. SimulateNavigation(GURL("http://www.google.es"), "es", true); EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_TRUE(GetTranslateInfoBar() != NULL); EXPECT_TRUE(CloseTranslateInfoBar()); // Let's switch to incognito mode, it should not be autotranslated in that // case either. TestingProfile* test_profile = static_cast(contents()->GetBrowserContext()); test_profile->set_incognito(true); SimulateNavigation(GURL("http://www.youtube.fr"), "fr", true); EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_TRUE(GetTranslateInfoBar() != NULL); EXPECT_TRUE(CloseTranslateInfoBar()); test_profile->set_incognito(false); // Get back to non incognito. // Now revert the always translate pref and make sure we go back to expected // behavior, which is show a "before translate" infobar. SetPrefObserverExpectation(TranslatePrefs::kPrefTranslateWhitelists); translate_prefs.RemoveLanguagePairFromWhitelist("fr", "en"); SimulateNavigation(GURL("http://www.google.fr"), "fr", true); EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::BEFORE_TRANSLATE, infobar->type()); } // Context menu. TEST_F(TranslateManagerTest, ContextMenu) { // Blacklist www.google.fr and French for translation. GURL url("http://www.google.fr"); Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); TranslatePrefs translate_prefs(profile->GetPrefs()); translate_prefs.BlacklistLanguage("fr"); translate_prefs.BlacklistSite(url.host()); EXPECT_TRUE(translate_prefs.IsLanguageBlacklisted("fr")); EXPECT_TRUE(translate_prefs.IsSiteBlacklisted(url.host())); // Simulate navigating to a page in French. The translate menu should show but // should only be enabled when the page language has been received. NavigateAndCommit(url); scoped_ptr menu( TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsItemPresent(IDC_CONTENT_CONTEXT_TRANSLATE)); EXPECT_FALSE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); // Simulate receiving the language. SimulateOnTranslateLanguageDetermined("fr", true); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsItemPresent(IDC_CONTENT_CONTEXT_TRANSLATE)); EXPECT_TRUE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); // Use the menu to translate the page. menu->ExecuteCommand(IDC_CONTENT_CONTEXT_TRANSLATE); // That should have triggered a translation. // The "translating..." infobar should be showing. TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::TRANSLATING, infobar->type()); // Simulate the translate script being retrieved. SimulateTranslateScriptURLFetch(true); int page_id = 0; std::string original_lang, target_lang; EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ("fr", original_lang); EXPECT_EQ("en", target_lang); process()->sink().ClearMessages(); // This should also have reverted the blacklisting of this site and language. EXPECT_FALSE(translate_prefs.IsLanguageBlacklisted("fr")); EXPECT_FALSE(translate_prefs.IsSiteBlacklisted(url.host())); // Let's simulate the page being translated. rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // The translate menu should now be disabled. menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsItemPresent(IDC_CONTENT_CONTEXT_TRANSLATE)); EXPECT_FALSE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); // Test that selecting translate in the context menu WHILE the page is being // translated does nothing (this could happen if autotranslate kicks-in and // the user selects the menu while the translation is being performed). SimulateNavigation(GURL("http://www.google.es"), "es", true); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); infobar->Translate(); EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); process()->sink().ClearMessages(); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); menu->ExecuteCommand(IDC_CONTENT_CONTEXT_TRANSLATE); // No message expected since the translation should have been ignored. EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); // Now test that selecting translate in the context menu AFTER the page has // been translated does nothing. SimulateNavigation(GURL("http://www.google.de"), "de", true); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); infobar->Translate(); EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); process()->sink().ClearMessages(); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "de", "en", TranslateErrors::NONE)); menu->ExecuteCommand(IDC_CONTENT_CONTEXT_TRANSLATE); // No message expected since the translation should have been ignored. EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); // Test that the translate context menu is enabled when the page is in an // unknown language. SimulateNavigation(url, "und", true); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsItemPresent(IDC_CONTENT_CONTEXT_TRANSLATE)); EXPECT_TRUE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); // Test that the translate context menu is disabled when the page is in an // unsupported language. SimulateNavigation(url, "qbz", true); menu.reset(TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsItemPresent(IDC_CONTENT_CONTEXT_TRANSLATE)); EXPECT_FALSE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); } // Tests that an extra always/never translate button is shown on the "before // translate" infobar when the translation is accepted/declined 3 times, // only when not in incognito mode. TEST_F(TranslateManagerTest, BeforeTranslateExtraButtons) { Profile* profile = Profile::FromBrowserContext(contents()->GetBrowserContext()); TranslatePrefs translate_prefs(profile->GetPrefs()); translate_prefs.ResetTranslationAcceptedCount("fr"); translate_prefs.ResetTranslationDeniedCount("fr"); translate_prefs.ResetTranslationAcceptedCount("de"); translate_prefs.ResetTranslationDeniedCount("de"); // We'll do 4 times in incognito mode first to make sure the button is not // shown in that case, then 4 times in normal mode. TranslateInfoBarDelegate* infobar; TestingProfile* test_profile = static_cast(contents()->GetBrowserContext()); test_profile->CreateExtensionProcessManager(); test_profile->set_incognito(true); for (int i = 0; i < 8; ++i) { SCOPED_TRACE(::testing::Message() << "Iteration " << i << " incognito mode=" << test_profile->IsOffTheRecord()); SimulateNavigation(GURL("http://www.google.fr"), "fr", true); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::BEFORE_TRANSLATE, infobar->type()); if (i < 7) { EXPECT_FALSE(infobar->ShouldShowAlwaysTranslateButton()); infobar->Translate(); process()->sink().ClearMessages(); } else { EXPECT_TRUE(infobar->ShouldShowAlwaysTranslateButton()); } if (i == 3) test_profile->set_incognito(false); } // Simulate the user pressing "Always translate French". infobar->AlwaysTranslatePageLanguage(); EXPECT_TRUE(translate_prefs.IsLanguagePairWhitelisted("fr", "en")); // Simulate the translate script being retrieved (it only needs to be done // once in the test as it is cached). SimulateTranslateScriptURLFetch(true); // That should have triggered a page translate. int page_id = 0; std::string original_lang, target_lang; EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); process()->sink().ClearMessages(); // Now test that declining the translation causes a "never translate" button // to be shown (in non incognito mode only). test_profile->set_incognito(true); for (int i = 0; i < 8; ++i) { SCOPED_TRACE(::testing::Message() << "Iteration " << i << " incognito mode=" << test_profile->IsOffTheRecord()); SimulateNavigation(GURL("http://www.google.de"), "de", true); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); EXPECT_EQ(TranslateInfoBarDelegate::BEFORE_TRANSLATE, infobar->type()); if (i < 7) { EXPECT_FALSE(infobar->ShouldShowNeverTranslateButton()); infobar->TranslationDeclined(); } else { EXPECT_TRUE(infobar->ShouldShowNeverTranslateButton()); } if (i == 3) test_profile->set_incognito(false); } // Simulate the user pressing "Never translate French". infobar->NeverTranslatePageLanguage(); EXPECT_TRUE(translate_prefs.IsLanguageBlacklisted("de")); // No translation should have occured and the infobar should be gone. EXPECT_FALSE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); process()->sink().ClearMessages(); ASSERT_TRUE(GetTranslateInfoBar() == NULL); } // Tests that we don't show a translate infobar when a page instructs that it // should not be translated. TEST_F(TranslateManagerTest, NonTranslatablePage) { // Simulate navigating to a page. SimulateNavigation(GURL("http://mail.google.fr"), "fr", false); // We should not have an infobar. EXPECT_TRUE(GetTranslateInfoBar() == NULL); // The context menu should be disabled. scoped_ptr menu( TestRenderViewContextMenu::CreateContextMenu(contents())); menu->Init(); EXPECT_TRUE(menu->IsItemPresent(IDC_CONTENT_CONTEXT_TRANSLATE)); EXPECT_FALSE(menu->IsCommandIdEnabled(IDC_CONTENT_CONTEXT_TRANSLATE)); } // Tests that the script is expired and refetched as expected. TEST_F(TranslateManagerTest, ScriptExpires) { ExpireTranslateScriptImmediately(); // Simulate navigating to a page and translating it. SimulateNavigation(GURL("http://www.google.fr"), "fr", true); TranslateInfoBarDelegate* infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); process()->sink().ClearMessages(); infobar->Translate(); SimulateTranslateScriptURLFetch(true); rvh()->TestOnMessageReceived(ChromeViewHostMsg_PageTranslated( 0, 0, "fr", "en", TranslateErrors::NONE)); // A task should have been posted to clear the script, run it. MessageLoop::current()->RunAllPending(); // Do another navigation and translation. SimulateNavigation(GURL("http://www.google.es"), "es", true); infobar = GetTranslateInfoBar(); ASSERT_TRUE(infobar != NULL); process()->sink().ClearMessages(); infobar->Translate(); // If we don't simulate the URL fetch, the TranslateManager should be waiting // for the script and no message should have been sent to the renderer. EXPECT_TRUE( process()->sink().GetFirstMessageMatching( ChromeViewMsg_TranslatePage::ID) == NULL); // Now simulate the URL fetch. SimulateTranslateScriptURLFetch(true); // Now the message should have been sent. int page_id = 0; std::string original_lang, target_lang; EXPECT_TRUE(GetTranslateMessage(&page_id, &original_lang, &target_lang)); EXPECT_EQ("es", original_lang); EXPECT_EQ("en", target_lang); }